Attach、Detach和DeleteObject objectcontext.attach

1.CWndAttatch和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 = newCWnd,执行是合法的,但是没有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对象,例如CEditedit; edit.create(...)并且此映射是永久的,知道此对象完蛋为止。如果用类似GetDlgItem函数也可以返回一个指针,并可以强制转换。GetDlgItem会到映射表里找。有2种映射表,一中是永久的,一种是临时的。直接用C++对象创建的WINDOWS对象或者是通过Attach的对象的映射关系都被放到永久表中,否则就在临时表中创建映射。所以GetDlgItem不推荐你保存返回的指针,因为你很难保证你的WINDOWS对象跟C++对象的关联是否放在永久表中。如果映射是放在临时表中,那么在空闲时间会被自动删除。用Attcah完全是为了方便用MFC类的成员函数去操纵WINDOWS对象。


2.CBitmapDetach和DeleteObject的关系

注意:当使用完资源后,必须通过调用函数以释放加速器表、位图、光标、图标以及菜单所占的内存资源;
加速器表:DesteoyAcceleratorTable;
位图:DeleteObject;光标:DestroyCursor;
Attach、Detach和DeleteObject objectcontext.attach

图标:Destroylcon;菜单:DestroyMenu


HBITMAP/CBitmap/BITMAP 三者之间的关系转换:
[cpp] view plaincopy
  1. HBITMAPhBitmap;
  2. CBitmapbitmap;
  3. BITMAPbm;
  4. bitmap.Attach(hBitmap);//由HBITMAP得到关联的CBitmap
  5. bitmap.GetBitmap(&bm);//由CBitmap得到关联的BITMAP
  6. 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的东西不是在栈里,而是在堆里。

  

爱华网本文地址 » http://www.aihuau.com/a/25101017/358848.html

更多阅读

柠檬和蜂蜜的做法、作用和功效 蜂蜜柠檬水功效

柠檬和蜂蜜的做法、作用和功效——简介柠檬和蜂蜜是很好的美容圣品,不管是外敷还是内服,都能起到显著的护肤功效。想要知道柠檬蜂蜜水的做法吗?不妨来看看小编讲解的小知识,详述柠檬和蜂蜜的作用和功效。柠檬和蜂蜜的做法、作用和功效

转载 C语言:随机函数rand()、srand()、random()和rando

原文地址:C语言:随机函数rand()、srand()、random()和randomized()的区别和用法作者:猎空声明一点:在VC++中,没有random()和randomize()函数,只有rand()和srand()函数。其中,random()和randomize()函数的使用的方法分别与rand()和srand()

扫盲|终于知道轩尼诗VS、VSOP和XO之间的区别了…… 轩尼诗vsop xo

不管你是走入烟酒行,或是夜场,或是大型商超,亦或是各大免税店,酒架上轩尼诗(Hennessy)的logo总是特别显眼。作为普通消费者,不少人对轩尼诗VS、VSOP和XO等系列是傻傻分不清楚。只道都是酒,为什么有这些区分?既是同一品牌,为什么价格相差悬殊?看

声明:《Attach、Detach和DeleteObject objectcontext.attach》为网友重获新的我分享!如侵犯到您的合法权益请联系我们删除