IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++11函数装饰器 -> 正文阅读

[C++知识库]C++11函数装饰器

C++11 函数装饰器

在pyhon里经常看到如下的代码

def Decorator(func):
    def _func(*args, **kwargs):
        print("before function")
        result = func(*args, **kwargs)
        print("after function")
        return result
    
    return _func


@Decorator
def func_a(a, b):
    print(a + b)


if __name__ == '__main__':
    func_a(1, 4)

运行上述代码,可以看到如下的结果

before function
5
after function

利用装饰器,我们可以实现函数计时、授权检查、日志等等复杂的功能。
那么如何在C++里实现一个类似的功能呢?
我们可以利用函数指针的功能,将函数暂存在装饰器类中,然后再调用前与调用后执行相应的功能。
话不多说直接上代码

/*!
 * @brief 装饰器基类
 * @tparam FUN_RET_TYPE 函数返回值类型
 * @tparam FUN_ARGS 函数参数类型
 */
template<typename FUN_RET_TYPE, typename ... FUN_ARGS>
class Decorator {
protected:
	//被装饰的函数
	std::function<FUN_RET_TYPE(FUN_ARGS...)> decorated_func;
public:
	explicit Decorator(const std::function<FUN_RET_TYPE(FUN_ARGS...)> &p_func) {
		decorated_func = p_func;
	}

	explicit Decorator(FUN_RET_TYPE(*p_func)(FUN_ARGS...)) :
			Decorator(std::function<FUN_RET_TYPE(FUN_ARGS...)>(p_func)) {
	}

	/*!
	 * 在这里实现装饰器的基本功能
	 * @param args
	 * @return
	 */
	virtual FUN_RET_TYPE operator()(FUN_ARGS...args) = 0;
};

这样我们就做好了一个简单的装饰器类,接下来我们将上面的python代码移植进来

/*!
 * 计时器
 * @tparam FUN_RET_TYPE
 * @tparam FUN_ARGS
 */
template<typename FUN_RET_TYPE, typename ... FUN_ARGS>
class FuncTimer : public Decorator<FUN_RET_TYPE, FUN_ARGS...> {
private:
	//计时器的提示字符串
	std::string tip_str;
public:
	explicit FuncTimer(const std::function<FUN_RET_TYPE(FUN_ARGS...)> &p_func, std::string tip_str = "")
			: Decorator<FUN_RET_TYPE, FUN_ARGS...>(p_func), tip_str(std::move(tip_str)) {
	}

	explicit FuncTimer(FUN_RET_TYPE(*p_func)(FUN_ARGS...), std::string tip_str = "")
			: Decorator<FUN_RET_TYPE, FUN_ARGS...>(p_func), tip_str(std::move(tip_str)) {
	}

	FUN_RET_TYPE operator()(FUN_ARGS...args) override {
		//使用函数前记录时间
		auto before = std::chrono::system_clock::now().time_since_epoch().count();
		//调用函数
		FUN_RET_TYPE result = this->decorated_func(args...);
		//调用函数后记录时间
		auto after = std::chrono::system_clock::now().time_since_epoch().count();
		//计算时间差并打印
		std::cout << tip_str << after - before << std::endl;
		return result;
	}
};

这里实现了一个简单的计时器装饰器,传入一个字符串用于控制打印的字符。
这里的装饰器有一个比较难以操作的问题:如果函数返回值为void,那么在模板替换的时候就会出现void result = this->decorated_func(args...);这段代码会直接报编译错误,因为C++并没有void型变量。相比C++而言,上面的那段python代码不会出错,因为python在处理无返回值函数时默认返回None,因此将None赋值给result不会出错。解决C++ void函数装饰问题可以简单的使用有返回值的lambda函数进行二次包装。这里直接使用宏定义的形式,如下:

#define CalFuncExecTime(content, text) FuncTimer<void *, void *>([&](void *_) -> void * {\
	do{content}while(false);\
	return nullptr;\
}, text)(nullptr)\

这里定义了返回值为void *的lambda函数,用装饰器包装了它,避免出错。同时这个宏还可以用来包装一段代码块。
下面是测试代码

class A {
public:
	int x;

	explicit A(int x) : x(x) {}

	int test(int a, int b) const {
		return a + b + x;
	}
};

int main() {
	FuncTimer<int, int, int> timer1(test, "timer1 : ");
	cout << timer1(2, 4) << std::endl;
	CalFuncExecTime(A a(6);
			                a.test(1, 4);,
	                "A : "
	);
}

输出:

timer1 : 5
6
A : 5

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-08-28 08:51:02  更:2021-08-28 08:51:17 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 16:58:19-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码