Go语言的defer语句是汤普森大神的神来之笔, 非常适合于局部资源安全的延迟释放. 本文将探讨如何在C++语言中实现一个近似的defer特性.
Go语言的defer具体用法可参考:
C++版defer
一直想在C++中也能模拟类似defer的效果, 经过无数次google和尝试之后重要找到解决方法. 整理后的代码在这里:
基本的使用方法:
FILE* fp = fopen("foo.txt", "rt");
if(fp == NULL) return false;
defer([&](){ printf("fclose(fp)\n"); fclose(fp); });
char* buf = new char[1024];
defer([&](){ printf("delete buf\n"); delete[] buf; });
defer([](){ printf("defer a: %d\n", __LINE__); });
defer([](){ printf("defer a: %d\n", __LINE__); });
defer([](){ printf("defer a: %d\n", __LINE__); });
{
defer([](){ printf("defer b: %d\n", __LINE__); });
defer([](){ printf("defer b: %d\n", __LINE__); });
defer([](){ printf("defer b: %d\n", __LINE__); });
}
defer([](){
printf("defer c:\n");
for(int i = 0; i < 3; ++i) {
defer([&](){ defer([&](){
printf("\ti = %d: begin\n", i);
defer([&](){ printf("\ti = %d\n", i); });
printf("\ti = %d: end\n", i);
});});
}
});
需要注意的地方:
- defer 定义的对象在超出作用域时执行闭包函数(析构函数)
- defer 定义的对象在同一个文件内部标识符不同(根据行号生成)
- defer 在全局作用域使用可能会出现重名现象(行号相同)
- defer 在判断语句使用可能提前执行(作用域结束时)
- defer 在循环语句内使用无效(作用域结束时)
- defer 和Go语言的defer并不完全等价