Unhandled exception jztsHook.exe:0x000005:Access Violation

yqy1234hf 2012-11-09 10:44:00
我写的是一个对话框程序,主要复制桌面上的某个窗口成图片,程序有两个按钮。
思路:
用HOOK返回鼠标在点击是的坐标点返回坐标点后用windowsfrompoint返回坐标对应的窗口句柄得到句柄后用getwindowrect获得窗口的坐标然后后就复制图片但是在图片分配内存的时候出错了
HBITMAP CJZTSHookDlg::CopyScreenToBitmap(LPRECT lpRect)
bool CJZTSHookDlg::SaveBitmapToFile(HBITMAP hBitmap, CString FileName)
这两个函数在单独调用的时候是正常的可以得到一副位图后来加了HOOK后就出现这个问题。请前辈纠错教我改下!
备注:错误调试的时候提示错误的地方:*lpbi = bi;但我不知道要怎么改。
主要代码:
_declspec(dllimport) void fhsbzb();
_declspec(dllimport) POINT zmzb();
void CJZTSHookDlg::OnButton1()
{
// TODO: Add your control notification handler code here
HWND hwnd=NULL;
LPRECT lpRect=NULL;
/*
fhsbzb();
point=zmzb();
*/
hwnd=::WindowFromPoint(point);
::GetWindowRect(hwnd,lpRect);
/////////////////////////////////////////////////////////////

//隐藏程序界面
ShowWindow(SW_HIDE);
//延时1s,使界面隐藏
Sleep(500);


HBITMAP hBitmap;
hBitmap = CopyScreenToBitmap(lpRect);
SaveBitmapToFile(hBitmap, "screen.bmp");

//显示程序界面
ShowWindow(SW_SHOW);

}

////////////////////////获得屏幕的图像///////////////////////////////////
HBITMAP CJZTSHookDlg::CopyScreenToBitmap(LPRECT lpRect)
{
HDC hScrDC,hMemDC; //屏幕和内存设备描述表
HBITMAP hBitmap,hOldBitmap; //位图句柄
int nX,nY,nX2,nY2; //选定区域坐标
int nWidth, nHeight; //位图宽度和高度
int xScrn, yScrn; //屏幕分辨率

//确保选定区域不为空矩形
if(IsRectEmpty(lpRect)) return NULL;

//为屏幕创建设备描述表
hScrDC=CreateDC("DISPLAY", NULL, NULL, NULL);

//为屏幕设备描述表创建兼容的内存设备描述表
hMemDC=CreateCompatibleDC(hScrDC);

//获得选定区域坐标
nX = lpRect->left;
nY = lpRect->top;
nX2 = lpRect->right;
nY2 = lpRect->bottom;

/*
nX = 3;
nY = 97;
nX2 = 76;
nY2 = 131;
*/

//获得屏幕分辨率
xScrn = GetDeviceCaps(hScrDC, HORZRES);
yScrn = GetDeviceCaps(hScrDC, VERTRES);
//确保选定区域是可见的
if(nX < 0)
{
nX = 0;
}
if(nY < 0)
{
nY = 0;
}
if(nX2 > xScrn)
{
nX2 = xScrn;
}

if(nY2 > yScrn)
{
nY2 = yScrn;
}
nWidth = nX2 - nX;
nHeight = nY2 - nY;

//创建一个与屏幕设备描述表兼容的位图
hBitmap=CreateCompatibleBitmap(hScrDC,nWidth,nHeight);
//把新位图选到内存设备描述表中
hOldBitmap=(HBITMAP)SelectObject(hMemDC, hBitmap);
//把屏幕设备描述表拷贝到内存设备描述表中
BitBlt(hMemDC,0,0,nWidth,nHeight,hScrDC,nX,nY,SRCCOPY);
//得到屏幕位图的句柄
hBitmap= (HBITMAP)SelectObject(hMemDC,hOldBitmap);
//清除
DeleteDC(hScrDC);
DeleteDC(hMemDC);
//返回位图句柄
return hBitmap;
}

//把HBITMAP保存成位图
bool CJZTSHookDlg::SaveBitmapToFile(HBITMAP hBitmap, CString FileName)
{

HDC hDC;

//当前分辨率下每象素所占字节数
int iBits;

//位图中每象素所占字节数
WORD wBitCount;

//定义调色板大小, 位图中像素字节大小 ,位图文件大小 , 写入文件字节数
DWORD dwPaletteSize=0, dwBmBitsSize=0, dwDIBSize=0, dwWritten=0;

//位图属性结构
BITMAP Bitmap;

//位图文件头结构
BITMAPFILEHEADER bmfHdr;

//位图信息头结构
BITMAPINFOHEADER bi;

//指向位图信息头结构
LPBITMAPINFOHEADER lpbi;

//定义文件,分配内存句柄,调色板句柄
HANDLE fh, hDib, hPal, hOldPal = NULL;

//计算位图文件每个像素所占字节数
hDC = CreateDC("DISPLAY", NULL, NULL, NULL);
iBits = GetDeviceCaps(hDC, BITSPIXEL) * GetDeviceCaps(hDC, PLANES);
DeleteDC(hDC);
if (iBits <= 1) wBitCount = 1;
else if (iBits <= 4) wBitCount = 4;
else if (iBits <= 8) wBitCount = 8;
else wBitCount = 24;

GetObject(hBitmap, sizeof(Bitmap), (LPSTR)&Bitmap);
bi.biSize = sizeof(BITMAPINFOHEADER);
bi.biWidth = Bitmap.bmWidth;
bi.biHeight = Bitmap.bmHeight;
bi.biPlanes = 1;
bi.biBitCount = wBitCount;
bi.biCompression = BI_RGB;
bi.biSizeImage = 0;
bi.biXPelsPerMeter = 0;
bi.biYPelsPerMeter = 0;
bi.biClrImportant = 0;
bi.biClrUsed = 0;

dwBmBitsSize = ((Bitmap.bmWidth * wBitCount + 31) / 32) * 4 * Bitmap.bmHeight;

//为位图内容分配内存
hDib = GlobalAlloc(GHND,dwBmBitsSize + dwPaletteSize + sizeof(BITMAPINFOHEADER));
lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDib);
*lpbi = bi;

// 处理调色板
hPal = GetStockObject(DEFAULT_PALETTE);
if (hPal)
{
hDC = ::GetDC(NULL);
hOldPal = ::SelectPalette(hDC, (HPALETTE)hPal, FALSE);
RealizePalette(hDC);
}

// 获取该调色板下新的像素值
GetDIBits(hDC, hBitmap, 0, (UINT) Bitmap.bmHeight, (LPSTR)lpbi + sizeof(BITMAPINFOHEADER)
+dwPaletteSize, (BITMAPINFO *)lpbi, DIB_RGB_COLORS);

//恢复调色板
if (hOldPal)
{
::SelectPalette(hDC, (HPALETTE)hOldPal, TRUE);
RealizePalette(hDC);
::ReleaseDC(NULL, hDC);
}

//创建位图文件
fh = CreateFile(FileName, GENERIC_WRITE,0, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);

if (fh == INVALID_HANDLE_VALUE) return FALSE;

// 设置位图文件头
bmfHdr.bfType = 0x4D42; // "BM"
dwDIBSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + dwPaletteSize + dwBmBitsSize;
bmfHdr.bfSize = dwDIBSize;
bmfHdr.bfReserved1 = 0;
bmfHdr.bfReserved2 = 0;
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER) + (DWORD)sizeof(BITMAPINFOHEADER) + dwPaletteSize;
// 写入位图文件头
WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
// 写入位图文件其余内容
WriteFile(fh, (LPSTR)lpbi, dwDIBSize, &dwWritten, NULL);
//清除
GlobalUnlock(hDib);
GlobalFree(hDib);
CloseHandle(fh);

return TRUE;
}

void CJZTSHookDlg::OnButton2()
{
// TODO: Add your control notification handler code here
fhsbzb();
point=zmzb();
}


/////////////////////////////////////////////////////////////////////////////////
钩子代码:


#include<windows.h>

HHOOK hook=NULL;
PMOUSEHOOKSTRUCT zbd=NULL;
POINT point;

LRESULT CALLBACK MouseProc(
int nCode, // hook code
WPARAM wParam, // message identifier
LPARAM lParam // mouse coordinates
)
{
zbd=(PMOUSEHOOKSTRUCT)lParam;
point=zbd->pt;
return 0;
}

void fhsbzb()
{
hook=SetWindowsHookEx(WH_MOUSE,MouseProc,GetModuleHandle("ZJHook"),0);
}

POINT zmzb()
{
return point;
}
...全文
176 3 打赏 收藏 转发到动态 举报
写回复
用AI写文章
3 条回复
切换为时间正序
请发表友善的回复…
发表回复
cjluxwd 2013-12-11
  • 打赏
  • 举报
回复
请问楼主怎么解决的,我也遇到同样的问题,希望楼主可以指教一下!
yqy1234hf 2012-12-19
  • 打赏
  • 举报
回复
虽然你没帮到我 不过还是谢谢你的回复 我自己解决了!
squall_qz 2012-11-10
  • 打赏
  • 举报
回复
是你运行的时候没点以管理员身份运行,还是这句 hook=SetWindowsHookEx(WH_MOUSE,MouseProc,GetModuleHandle("ZJHook"),0); ZJHOOK不知道是什么……

16,472

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Web++
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧