DeskBand实现之——遇到的问题2

1. 最近又发现一个问题,相当严重,但重现机率相当小,最后通过打LOG,调了半天,终于找到原因。

现象如下:
把Desk Band关闭,再重新打开就会把explorer弄挂掉。一定要多试几次才可能出现。
原因是什么呢?
Desk Band出来是会创建一个窗体的,同时也会注册一个全局类,调用 RegisterClass 函数就行。
RegisterClass会指定一个消息处理函数地址,也就是一个回调函数。
当你把DLL卸载后,这个全局类的信息还是保存在系统中,当DLL再次加载时,又会调用RegisterClass来注册一个类,系统发现这个类已经被注册,就会返回一个错误代码为ERROR_CLASS_ALREADY_EXISTS的错误,指示类已经被注册。如果此时忽略这个错误,创建窗体,如果运气好的话,没什么问题,会创建成功,如果运气不好的话,就会在CreateWindow(Ex)这里挂掉,因为当调用CreateWindow时,它会发送WM_NCCREATE消息到这个窗体对应的消息处理函数中,但系统类里面保存的处理函数的地址还是原来DLL的地址,在这个新加载进来的DLL里面去找这个地址,就有可能出错,因为这个地址所指的地方有可能根本不是这个回调函数。所以,我们在窗体销毁时一定要调用 UnregisterClass 函数来反注册这个系统类。
 

2. 如何立即让系统卸载不用的DLL

系统是不会马上去卸载那些已经可以卸载的DLL,它有一个轮循周期,据我观察,好像是15分钟左右吧。要想立即卸载不用的DLL,可以调用
CoFreeUnusedLibraries函数。

或者通过如下方式也可以做到:(针对Desk Band)

// Find handle to the task bar.
HWND hTaskbarWnd = FindWindow(L"Shell_TrayWnd", NULL);
// If task bar receives this message, it will call CoFreeUnusedLibraries function
// immediately to free unused libraries.
PostMessage(hTaskbarWnd, WM_TIMER, 24, 0);


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值