C++之RAII技术解析
2015-01-22 22:51
519 查看
1.什么是RAII 技术?
我们在C++中经常使用new申请了内存空间,但是却也经常忘记delete回收申请的空间,容易造成内存溢出,于是RAII技术就诞生了,来解决这样的问题。RAII(Resource Acquisition Is Initialization)机制是Bjarne Stroustrup首先提出的,是一种利用对象生命周期来控制程序资源(如内存、文件句柄、网络连接、互斥量等等)的简单技术。 我们知道在函数内部的一些成员是放置在栈空间上的,当函数返回时,这些栈上的局部变量就会立即释放空间,于是Bjarne Stroustrup就想到确保能运行资源释放代码的地方就是在这个程序段(栈)中放置的对象的析构函数了,因为stack
winding会保证它们的析构函数都会被执行。RAII就利用了栈里面的变量的这一特点。RAII 的一般做法是这样的:在对象构造时获取资源,接着控制对资源的访问使之在对象的生命周期内始终保持有效,最后在对象析构的时候释放资源。借此,我们实际上把管理一份资源的责任托管给了一个存放在栈空间上的局部对象。
这种做法有两大好处:
(1)不需要显式地释放资源。
(2)采用这种方式,对象所需的资源在其生命期内始终保持有效。
2.实战应用
2.1一个简单的例子:指针申请空间,释放空间
在很多时候,为了实现多线程之间的数据同步,我们会使用到 mutex,critical section,event,singal 等技术。但在使用过程中,由于各种原因,有时候,我们会遇到一个问题:由于忘记释放(Unlock)锁,产生死锁现象。
采用RAII 就可以很好的解决这个问题,使用着不必担心释放锁的问题. 示例代码如下:
My_scope_lock 为实现 局部锁的模板类.
LockType 抽象代表具体的锁类 .如基于 mutex 实现 mutex_lock 类.
//global vars
int counter = 0;
void routine();
mutex_lock m_mutex_lock;
void routine()
{
My_scope_lock l_lock(m_mutex_lock);
counter++;
...//others...
}
我们可以根据上面的例子类推出好多这样例子。如读写文件的时候很容易忘记关闭文件,如果借用 RAII技术,就可以规避这种错误。再如对数据库的访问,忘记断开数据库连接等等都可以借助RAII 技术也解决。
参考文章:
一个RAII工厂实现 http://www.codeproject.com/Articles/10141/RAII-Dynamic-Objects-and-Factories-in-C
RAII局部锁技术 http://www.cnblogs.com/zhangyunkui/archive/2009/11/13/1602514.html
RAII文件操作 http://www.cnblogs.com/gnuhpc/archive/2012/12/04/2802307.html http://blog.csdn.net/hunter8777/article/details/6327704
我们在C++中经常使用new申请了内存空间,但是却也经常忘记delete回收申请的空间,容易造成内存溢出,于是RAII技术就诞生了,来解决这样的问题。RAII(Resource Acquisition Is Initialization)机制是Bjarne Stroustrup首先提出的,是一种利用对象生命周期来控制程序资源(如内存、文件句柄、网络连接、互斥量等等)的简单技术。 我们知道在函数内部的一些成员是放置在栈空间上的,当函数返回时,这些栈上的局部变量就会立即释放空间,于是Bjarne Stroustrup就想到确保能运行资源释放代码的地方就是在这个程序段(栈)中放置的对象的析构函数了,因为stack
winding会保证它们的析构函数都会被执行。RAII就利用了栈里面的变量的这一特点。RAII 的一般做法是这样的:在对象构造时获取资源,接着控制对资源的访问使之在对象的生命周期内始终保持有效,最后在对象析构的时候释放资源。借此,我们实际上把管理一份资源的责任托管给了一个存放在栈空间上的局部对象。
这种做法有两大好处:
(1)不需要显式地释放资源。
(2)采用这种方式,对象所需的资源在其生命期内始终保持有效。
2.实战应用
2.1一个简单的例子:指针申请空间,释放空间
void Func() { int *ip = new int[10]; ...//operations ...//operations ...//operations delete[] ip;//if not free mem, memory overflow }使用RAII技术后:
template<class PointerType> class My_Pointer { public: My_Pointer(PointerType* _ptr, size_t sz) { _ptr = new PointerType[sz]; m_ptr = _ptr; } ~My_Pointer() { delete []m_ptr; } protected: PointerType m_ptr; }2.2 scope lock (局部锁技术)
在很多时候,为了实现多线程之间的数据同步,我们会使用到 mutex,critical section,event,singal 等技术。但在使用过程中,由于各种原因,有时候,我们会遇到一个问题:由于忘记释放(Unlock)锁,产生死锁现象。
采用RAII 就可以很好的解决这个问题,使用着不必担心释放锁的问题. 示例代码如下:
My_scope_lock 为实现 局部锁的模板类.
LockType 抽象代表具体的锁类 .如基于 mutex 实现 mutex_lock 类.
template<class LockType> class My_scope_lock { public: My_scope_lock(LockType& _lock):m_lock(_lock) { m_lock.occupy(); } ~My_scope_lock() { m_lock.relase(); } protected: LockType m_lock; }使用的时候:
//global vars
int counter = 0;
void routine();
mutex_lock m_mutex_lock;
void routine()
{
My_scope_lock l_lock(m_mutex_lock);
counter++;
...//others...
}
我们可以根据上面的例子类推出好多这样例子。如读写文件的时候很容易忘记关闭文件,如果借用 RAII技术,就可以规避这种错误。再如对数据库的访问,忘记断开数据库连接等等都可以借助RAII 技术也解决。
参考文章:
一个RAII工厂实现 http://www.codeproject.com/Articles/10141/RAII-Dynamic-Objects-and-Factories-in-C
RAII局部锁技术 http://www.cnblogs.com/zhangyunkui/archive/2009/11/13/1602514.html
RAII文件操作 http://www.cnblogs.com/gnuhpc/archive/2012/12/04/2802307.html http://blog.csdn.net/hunter8777/article/details/6327704
相关文章推荐
- C++之RAII技术解析
- 利用C++ RAII技术自动回收堆内存
- 全面解析查找技术数据结构第7章c++版
- C++ : 应用 RAII 技术在 Windows 下实现自动释放锁
- Caffemodel数据结构解析与Protocol Buffer技术详解(C++实例)
- C++的try块与异常处理及调试技术实例解析
- C++中利用析构避免资源泄露(RAII技术)
- 微软软件实现技术授课系列内容之三:Project in C++ Coding Practice
- C++中最希望出现得技术
- 四种 XML 解析技术之不完全测试
- 解析.NET框架下的数据类型转化技术
- IPv6任播技术及其任播地址解析协议概述
- C++虚函数调用的反汇编解析
- 解析XML与面向Web的数据挖掘技术
- 永恒与瞬间之美(C++技术小品文)
- 解析.Net框架下的XML编程技术
- 解析新一代WLAN安全技术IEEE 802.11i、WPA和WAPI
- Effective STL 条款6 : 当心C++另人迷惑的解析
- 关于c++名字解析规则的一次小研究,
- Java中四种XML解析技术之不完全测试(转载)