#ifdef _DEBUG
void CHandleMap::RemoveHandle(HANDLE h)
{
// make sure the handle entry is consistent before deleting
CObject* pTemp = LookupTemporary(h);
if (pTemp != NULL)
{
// temporary objects must have correct handle values
HANDLE* ph = (HANDLE*)((BYTE*)pTemp + m_nOffset);  // after CObject
ASSERT(ph[0] == h || ph[0] == NULL);
if (m_nHandles == 2)
ASSERT(ph[1] == h);  // 报错定位此处
}
pTemp = LookupPermanent(h);
if (pTemp != NULL)
{
HANDLE* ph = (HANDLE*)((BYTE*)pTemp + m_nOffset);  // after CObject
ASSERT(ph[0] == h);
// permanent object may have secondary handles that are different
}
// remove only from permanent map -- temporary objects are removed
//  at idle in CHandleMap::DeleteTemp, always!
m_permanentMap.RemoveKey((LPVOID)h);
}
#endif
程序编译通过,运行大部分时间是正常状态的!!!可是时而会出现致命的error,但是又不是经常报错。
上网查资料,没有解释比较清楚的文章,所以在这里发布一下,向高手们求助!!!
我的工程是在WinCE下开发的,其中是有加载位图,使用了多线程,也有操纵数据库等方面操作。具体这种情况不是很清楚是哪里的问题,请问使用什么方法去跟踪一下呢?

解决方案 »

  1.   

    调试看看是不是重复Remove了,或者句柄值不对,或是有内存越界等破坏了CHandleMap。
      

  2.   

    MFC类最好不要跨线程使用,尤其不要跨线程分配和销毁。
      

  3.   

    我使用线程只做大量的后台数据库初始化和部分后台计算。1楼说的那个情况,请问用什么办法去实现跟踪!!!我每切换一个换面都会调用一次!!!
    我跟踪都是直接跳到断点处,不知道,
     ASSERT(ph[1] == h);  // 报错定位此处
    这个前面的那个代码端在哪里??
      

  4.   

    可以在Remove时把有关信息写入日志文件以便查看。
      

  5.   

    用GetDC()得到的DC, 必须调用ReleaseDC()
    用CreateDC()创建的DC, 必须调用DeleteDC()两者是不能混淆的.
    一种典型的错误代码如下:
    CDC* pDC = GetDC();
    …..
    //做一些无聊的事
    …..pDC->DeleteDC(); //做了错误的事一般情况下, 上面这个代码也能够工作, 就是有时要报一点
    CHandleMap::RemoveHandle(HANDLE h)异常, 跟踪进代码也看不出啥问题.
    区别:
    ReleaseDC()仅是释放对DC的引用
    DeleteDC()是删除DC对象, 使用GetDC()的时候, 明显的这个DC不是我们自己创建的, 使用DeleteDC()的话就会将别人(通常是MFC框架)创建的DC干掉, 这样当别人要使用自己创建的DC时, 就会产生异常 了.ReleaseDC()恰好解决这个问题, 它仅是释放一个引用, 告诉创建者DC, 我已经不用了.