C++萃取技术的一个简单初探
首先本文并不是讲解C++萃取技术,关于C++的萃取技术网上有很多文章,推荐http://www.cppblog.com/woaidongmao/archive/2008/11/09/66387.html如果这篇文章能看懂肯定也能看懂我这篇小博文了。
熟悉C++的人肯定都用过auto_ptr这个智能指针,其原理就是在auto_ptr构造的时候将申请到的内存地址放进该对象的内部保存,而这个对象析构的时候则调用delete释放之前申请的内存,这样不仅仅省去了程序员时刻关注何时释放内存的顾忌,同时即使是程序在new和delete中间发生异常被捕获内存仍然会释放不会有泄漏。同时也可以做引申,可以自己写一个类,里面保存临界区对象,当对象创建时进入临界区,而对象析构的时候自从释放临界区,这样也可以保证进临界区和出临界区中间发生异常后锁仍然会释放,在一部分情况下保证程序不会有太大的异常。相关伪代码如下:
1 class CAutoLock 2 { 3 public: 4 CAutoLock(LPCRITICAL_SECTION lpCriticalSection) 5 { 6 m_lpCriticalSection = lpCriticalSection; 7 EnterCriticalSection(m_lpCriticalSection); 8 } 9 ~CAutoLock(){LeaveCriticalSection(m_lpCriticalSection); } 10 private: 11 LPCRITICAL_SECTION m_lpCriticalSection; 12 };
好了,锁和智能指针都有了,但是如果想类似让一个文件句柄HANDLE一样在遇到return时关闭掉,该怎么做呢?其实自己实现一个类似auto_ptr,将内存地址换成文件句柄就可以了,但是如果又遇到一个SOCKET呢?其实可以用模板偏特化搞定。但是我现在介绍一种利用C++萃取技术,用一个主体类再加上一些辅助类完成以上功能。其原理也很简单,利用C++萃取技术,自动识别需要释放的数据类型,然后根据不同的数据类型选择不同的释放方式。相关代码如下:
1 #pragma once 2 #include <windows.h> 3 #include <Winsock2.h> 4 5 #pragma comment(lib,"Ws2_32.lib") 6 7 /************************************************* 8 示例程序能够识别的类型: 9 第一为基础句柄(为用CloseHandle释放的) 10 第二个为网络通信SOCKET,需要用closesocket释放 11 可以根据实际需要自己添加实现,例子程序就只实现这两个功能 12 *************************************************/ 13 14 //基础句柄类(CloseHandle释放类型) 15 class CCommomHandleObject 16 { 17 public: 18 void clear(HANDLE hCommhandle) 19 { 20 if (INVALID_HANDLE_VALUE != hCommhandle) 21 { 22 CloseHandle(hCommhandle); 23 } 24 } 25 }; 26 27 //套接字类(closesocket释放类型) 28 class CSOCKETObject 29 { 30 public: 31 void clear(SOCKET m_hSocket) 32 { 33 if (INVALID_SOCKET != m_hSocket) 34 { 35 closesocket(m_hSocket); 36 } 37 } 38 }; 39 40 //C++萃取基础类 41 template<class _xTypeShow> 42 class CTraitsBaseObject 43 { 44 }; 45 46 //根据HANDLE识别出CCommomHandleObject类 47 template<> 48 class CTraitsBaseObject<HANDLE> 49 { 50 public: 51 typedef CCommomHandleObject _data_type_show; 52 }; 53 54 //根据SOCKET识别出CSOCKETObject 55 template<> 56 class CTraitsBaseObject<SOCKET> 57 { 58 public: 59 typedef CSOCKETObject _data_type_show; 60 }; 61 /**************************************************************************** 62 资源释放类,本例的重点,需要根据每个须释放的资源绑定一个CContainerClass对象 63 然后对象析构时会根据CTraitsBaseObject<_xType>::_data_type_show识别出是否为 64 自己认识的数据类型,如不认识则直接报错,然后创建临时对象调用clear, 65 根据实际情况销毁绑定的资源。 66 ****************************************************************************/ 67 template<class _xType> 68 class CContainerClass 69 { 70 public: 71 CContainerClass(_xType &_tay ){m_hTay = _tay;} 72 ~CContainerClass(void){clear();} 73 private: 74 CContainerClass(){} 75 CContainerClass(CContainerClass &tay){} 76 77 void clear() 78 { 79 CTraitsBaseObject<_xType>::_data_type_show().clear(m_hTay); 80 } 81 private: 82 _xType m_hTay; 83 };
以上代码只是完成了需要用CloseHandle释放的句柄和SOCKET两种资源类型,需要增加时只需增加相信的类型即可。而资源申请后创建一个CContainerClass对象,选择对应的类型将两者绑定即可。实际应用代码如下:
1 int _tmain(int argc, _TCHAR* argv[]) 2 { 3 HANDLE hFileFile = CreateFile(_T("D:\\Program Files\\小软件\\jpskb.exe"), GENERIC_READ | GENERIC_WRITE, 4 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL); 5 CContainerClass<HANDLE> hContainerClassFile(hFileFile); 6 7 SOCKET sock = INVALID_SOCKET; 8 CContainerClass<SOCKET> sContainerClasssock(sock); 9 10 int i = 0; 11 //CContainerClass<int> iCContainerClassi(i); //int为不识别类型,编译不过 12 return 0; 13 }
本来想过需不需要把不识别的类型做成一个空的clear函数,就是遇到不认识的类型什么都不做直接跳过。但是后来想想要是这样如果遇到没有对应的资源销毁类恐怕会有内存泄漏,于是让程序遇到不认识的类型直接编译不过即可。
其实写这篇文章有点多此一举,因为实际开发中根本不需要这么做。但是萃取技术作为C++当中一个比较难懂的知识点,如果能设计一个实际用途方案对深入理解还是很有帮助的,所以我感觉并非画蛇添足了。