转载:Attatch和Detach的关系
来源:互联网 发布:java面向 编辑:程序博客网 时间:2024/04/28 03:53
下面的文章中提到的有一点很好:
“Detach把这个“脐带”剪断,析构的时候就不会销毁”。
这个是在我们开发的过程中经常会犯的一个错误,像使用CBitmap时就经常忘记Detach,结果图片显示不出来。
转载地址:http://blog.sina.com.cn/s/blog_86fe5b440101au9o.html
1.CWnd Attatch和Detach的关系
首先,要明白Windows对象和MFC对象的区别。MFC对象实际上并没有把整个Windows对象都包装在其中,它只是有一个窗口句柄而已,这个窗口句柄如果指向一个实际存在的窗口对象(窗口对象,也就是WNDCLASS,是一个Windows对象),那么这个MFC对象就是有效的,否则这个MFC对象是空的。如果你还不明白,请回忆一下,当我们使用MFC创建一个窗口时,是分两步进行的,第一步,new一个CWnd对象,这一步是创建MFC对象,但是其中的HWND还是非法的,因为对应的Windows对象还没有被创建出来;第二步,调用CWnd的成员函数Create创建真正的Windows对象,同时,把先前创建的MFC的CWnd对象的HWND成员指向该窗口,这样才算创建完毕一个窗口。而如果你是用SDK方式,那么只要创建一个 WNDCLASS结构,然后调用Create或者CreateEx就创建了一个窗口。
好,现在回答你的问题,你可以假设,现在你已经有了一个有效窗口句柄,那么你想把这个窗口和一个CWnd对象关联起来怎么办?很简单,用Attach,其实就是让一个CWnd对象的HWND成员指向这个窗口句柄。这就是Attach主要完成的任务。
第二个,关于Detach。如前所述,WNDCLASS其实和CWnd根本没有什么关系。它们之间只是通过CWnd的成员HWND联系起来的。如果把 Attach看做“联姻”的话,那么Detach就是“离婚”了,通俗地说,就是切断一个CWnd对象和一个有效窗口的脐带。为什么要切断呢?因为 CWnd是C++的对象,C++的对象有一个生存期的概念,脱离了该对象的作用域,这个对象就要被销毁,但是Windows对象没有这个特点,当销毁 CWnd对象的时候,我们不一定希望WNDCLASS一起被销毁,那么在此之前,我们就先要把这个“脐带”剪断,以免“城门失火,殃及池鱼”。
Detach之后CMenu就释放了HMENU的控制权,析构的时候就不会销毁菜单句柄。
CWnd,CDC, Cxxx等都是MFC的类,这些类提供了很多成员函数来执行系统调用等操作,但是核心的类成员数据都是句柄,(包括窗口句柄,DC句柄,线程句柄等)。
m_hWnd,m_hDC,m_hThread如果这些类对象的这些句柄为空,就表示无效对象。比如你 CWnd * pWnd = new CWnd,执行是合法的,但是没有Create就没有窗口句柄,基本上就是一个无效对象,只是系统内存申请了内存分配空间而已,很多操作都无法执行(debug模式下会出现断言错误窗口)。
但是,假如你申请了一个CWnd,CDC的对象实体,可以使用Attach来指定一个有效的句柄付给这个对象。那么此对象就是Valid的了。你可能跟踪看看Attach的实现,其实就是给m_hxxx赋值,而Detach就是值复位。
这2个操作没有申请内存释放内存操作,就是一个赋值而已,是不是成对使用看你代码的实际情况吧。只要理解了这2函数的意义就行了
其实CWnd,CDC等类就是多了一个包装,方便用户(程序开发者)使用,如果用SDK开发,完全可以实现一样的功能。
句柄是操作系统内核对象,而窗口指针、DC指针是用户对象(由你的程序管理)。
延伸理解下Attach/Detach:
Attach是把一个C++对象与一个WINDOWS对象关联,直到用detach则把关联去掉。如果Attach了以后没有Detach,则C++对象销毁的时候WINDOWS对象跟着一起完蛋。Attach了以后,C++对象的指针WINDOWS对象的HWND会有一个映射关系,其作用相当于你直接用一个C++对象去Create一个WINDOWS对象,例如 CEdit edit; edit.create(...) 并且此映射是永久的,知道此对象完蛋为止。如果用类似GetDlgItem函数也可以返回一个指针,并可以强制转换。GetDlgItem会到映射表里找。有2种映射表,一中是永久的,一种是临时的。直接用C++对象创建的WINDOWS对象或者是通过Attach的对象的映射关系都被放到永久表中,否则就在临时表中创建映射。所以GetDlgItem不推荐你保存返回的指针,因为你很难保证你的WINDOWS对象跟C++对象的关联是否放在永久表中。如果映射是放在临时表中,那么在空闲时间会被自动删除。用Attcah完全是为了方便用MFC类的成员函数去操纵WINDOWS对象。
2.CBitmap Detach和DeleteObject的关系
注意:当使用完资源后,必须通过调用函数以释放加速器表、位图、光标、图标以及菜单所占的内存资源;
加速器表:DesteoyAcceleratorTable;
位图:DeleteObject;光标:DestroyCursor;
图标:Destroylcon;菜单:DestroyMenu
HBITMAP/CBitmap/BITMAP 三者之间的关系转换:
[cpp] view plaincopy
HBITMAP hBitmap;
CBitmap bitmap;
BITMAP bm;
bitmap.Attach(hBitmap);//由HBITMAP 得到关联的CBitmap
bitmap.GetBitmap(&bm); // 由CBitmap 得到关联的BITMAP
hBitmap=(HBITMAP)bitmap.GetSafeHandle();//由CBitmap得到相关的HBITMAP
一些通过FromHandle()或者Create...()获得的指针需要delete吗?
我知道很多gdi对象在Create后需要使用DeleteObject()释放其句柄,但是否需要delete呢?
我给你说三种情况,但我们首先做一个假设,就是MFC封装的GDI类在析构时没有做任何动作,也就是说,它是个纯粹的“简单封装”,那么:
1.pBmp->Detach将使C++的对象与GDI对象分离开来,但二者都没有释放。此时必须分别用deletepBmp和DeleteObject将二者分别释放;
2.pBmp->DeleteObject将使GDI对象被释放,而C++对象本身不会释放。你可以用Attach重新使其与某个GDI对象关联,或者,用delete将其释放;
3.deletepBmp(注意,我们假定析构时不调用DeleteObject)将使C++对象消亡,而对应的GDI对象依然存在。要使GDI对象释放,必须再次调用DeleteObject。
注意上面说的第三种情况,在实际的MFC实现中,为了简化程序员的负担,在C++对象析构时,与之关联的GDI对象也会释放。我之所以这样假定,是为了让你明白C++对象与GDI对象的区别。
还有,new的东西不是在栈里,而是在堆里。
- 转载:Attatch和Detach的关系
- thread的join和detach
- 线程的join和detach
- 关于attach和detach的疑问
- Attach和Detach的简单说明
- 线程的join 和detach状态
- pthread join 和detach 的处理
- thread线程的join和detach
- jQuery之detach()和remove()的区别
- 线程的join 和detach状态
- thread中join和detach的区别
- jquery detach()和remove()的区别
- thread中join和detach的区别
- thread中join和detach的区别
- 转载:scanf和缓冲区的关系
- 关于Hibernate和JPA的关系<转载!>
- 转载:JavaScript和DOM的关系
- 【转载HTML和jsp的关系】
- 快速目录和文件遍历
- 软件工程(三)工程流程
- bootstrap的text-muted和text-danger
- 使用SharpPCap在C#下进行网络抓包
- 原型设计模式
- 转载:Attatch和Detach的关系
- 程序员学习网站大全
- oracle怎样修改表名、列名、字段类型、添加表列、删除表列
- 计算机端口
- 剑指Offer:面试题01---赋值运算符函数
- 统计大量文本中重复字符串的最大个数
- 数据挖掘(四)聚类
- jsp和servlet之间传值问题
- SUSE Enterprise Server 设置防火墙