为什么RegisterClass会失败

hxh76ahei 2006-02-17 11:22:12
各位大佬好!请教如下问题。我的程序要Skype通讯,利用Skype发的SDK中的例子做了小许改动,在win2000和winxp上都没问问题,可程序在win98上却失败了。如下:
void main(void)
{
// create window class
// create dummy/hidden window for processing messages
// run message loop thread
// do application control until exit
// exit: send QUIT message to our own window
// wait until thred terminates
// destroy main window
// destroy window class
uiGlobal_MsgID_SkypeControlAPIAttach=RegisterWindowMessage("SkypeControlAPIAttach");
uiGlobal_MsgID_SkypeControlAPIDiscover=RegisterWindowMessage("SkypeControlAPIDiscover");
if( uiGlobal_MsgID_SkypeControlAPIAttach!=0 && uiGlobal_MsgID_SkypeControlAPIDiscover!=0 )
{
printf("STEP 1\n");
if( Initialize_CreateWindowClass() )
{
printf("STEP 2\n");
if( Initialize_CreateMainWindow() )
{
printf("STEP 3\n");
hGlobal_ThreadShutdownEvent=CreateEvent( NULL, TRUE, FALSE, NULL);
if( hGlobal_ThreadShutdownEvent!=NULL )
{
printf("STEP 4\n");
if( _beginthread( &Global_InputProcessingThread, 64*1024, NULL)!=(unsigned long)-1 )
{
printf("STEP 5\n");
Global_MessageLoop();
Global_Console_CancelReadRow();
WaitForSingleObject( hGlobal_ThreadShutdownEvent, INFINITE);
}
CloseHandle(hGlobal_ThreadShutdownEvent);
}
DeInitialize_DestroyMainWindow();
}
DeInitialize_DestroyWindowClass();
}
}
}
在第一步中bool Initialize_CreateWindowClass(void)
{
unsigned char *paucUUIDString;
RPC_STATUS lUUIDResult;
bool fReturnStatus;
UUID oUUID;

fReturnStatus=false;
lUUIDResult=UuidCreate(&oUUID);
hInit_ProcessHandle=(HINSTANCE)OpenProcess( PROCESS_DUP_HANDLE, FALSE, GetCurrentProcessId());
if( hInit_ProcessHandle!=NULL && (lUUIDResult==RPC_S_OK || lUUIDResult==RPC_S_UUID_LOCAL_ONLY) )
{
Printf("open process ok\n");
if( UuidToString( &oUUID, &paucUUIDString)==RPC_S_OK )
{
Printf("uuid converse ok\n");
WNDCLASS oWindowClass;

strcpy( acInit_WindowClassName, "Skype-API-Test-");
strcat( acInit_WindowClassName, (char *)paucUUIDString);

oWindowClass.style=CS_HREDRAW|CS_VREDRAW|CS_DBLCLKS;
oWindowClass.lpfnWndProc=(WNDPROC)&SkypeAPITest_Windows_WindowProc;
oWindowClass.cbClsExtra=0;
oWindowClass.cbWndExtra=0;
oWindowClass.hInstance=hInit_ProcessHandle;
oWindowClass.hIcon=NULL;
oWindowClass.hCursor=NULL;
oWindowClass.hbrBackground=NULL;
oWindowClass.lpszMenuName=NULL;
oWindowClass.lpszClassName=acInit_WindowClassName;

if( RegisterClass(&oWindowClass)!=0 )
{
fReturnStatus=true;
}
else
{
printf("register wndclass failure!!!\n");
}



RpcStringFree(&paucUUIDString);
}
}
if( fReturnStatus==false )
CloseHandle(hInit_ProcessHandle),hInit_ProcessHandle=NULL;
return(fReturnStatus);
}
在win98上运行时输出:
step 1
open process ok
uuid converse ok
register wndClass failure
我把上面oWindowClass.hInstance=hInit_ProcessHandle;改为
oWindowClass.hInstance=NULL;后再输出就是:
step 1
open process ok
uuid converse ok
step 2
上面那个线程是要处理输入输出的,所以不能没有,win98也是支持多线程的,我的程序为什么就跑不起来呢?请给点指。
...全文
284 6 打赏 收藏 转发到动态 举报
写回复
用AI写文章
6 条回复
切换为时间正序
请发表友善的回复…
发表回复
wangk 2006-02-17
  • 打赏
  • 举报
回复
晕,HINSTANCE不是内核对象,不能传openprocess打开得句柄,应该用GetModuleHandle(NULL)返回的值。
hxh76ahei 2006-02-17
  • 打赏
  • 举报
回复
to ydfivy:多谢大佬回复,可我看msdn上,上面的api好象98都支持的。那个注册过程也是很常见的呀。难道就是不能挂进去那个线程,可是在win32的application中都有wndclass.hInstance = hCurrentInst; 区别不就是这个hCurrentInst是糸统传进去的,而我那个用openprocess打开了一个吗?
一个傻冒 2006-02-17
  • 打赏
  • 举报
回复
98下.有些API可能不支持,或者不能使用的.
hxh76ahei 2006-02-17
  • 打赏
  • 举报
回复
to wangk;好心人,我按你说的又改了一下,在98下编还是失败了。再说一次同样方法在win2000和winxp上是没有问题的。错在哪 儿了?
wangk 2006-02-17
  • 打赏
  • 举报
回复
GetModuleHandle(NULL)返回的是HMODULE,也要强制转HINSTANCE。
hxh76ahei 2006-02-17
  • 打赏
  • 举报
回复
to wangk:多谢有两颗红星的大佬,在上面的程序中hInit_ProcessHandle=(HINSTANCE)OpenProcess( PROCESS_DUP_HANDLE, FALSE, GetCurrentProcessId());打开了一个存在的线程共享它的句柄并进行HINSTANCE强制转换,在编绎中并没有错,而且有open process ok的输出,同时我用你的GetModuleHandle(NULL)同样失败了。这个程序是一个console模式的,编绎生成的程序在win2000和winxp上都有正确结果。但在98上出现了上述错误。
有点小问题,颜色不能交替变换 #include #include #include LRESULT CALLBACK WinProc( HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam ); int WINAPI WinMain( HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd ) { //定义一个窗口类 WNDCLASS wincls; wincls.cbClsExtra=0; wincls.cbWndExtra=0; wincls.hbrBackground=(HBRUSH)GetStockObject(WHITE_BRUSH); wincls.hCursor=LoadCursor(NULL,IDC_CROSS); wincls.hIcon=LoadIcon(NULL,IDI_ERROR); wincls.lpfnWndProc=WinProc; wincls.lpszClassName="Lucky_wei"; wincls.lpszMenuName=NULL; wincls.hInstance=hInstance; wincls.style=CS_HREDRAW | CS_VREDRAW; if(!RegisterClass(&wincls;)) { MessageBox(NULL,"窗口类注册失败","注册窗口",NULL); return 1; } // //创建窗口 ,定义一个变量用来保存成功建立的窗口的句柄 HWND hwnd; hwnd=CreateWindow("Lucky_wei","旋转小风车",WS_OVERLAPPEDWINDOW,0,0,600,600,NULL,NULL,hInstance,NULL); if(!hwnd) { MessageBox(NULL,"创建窗口失败","创建窗口",NULL); return 1; } // // //显示并更新窗口 ShowWindow(hwnd,SW_SHOWNORMAL); UpdateWindow(hwnd); // // //定义消息结构体,实现消息循环 MSG msg; while(GetMessage(&msg;,NULL,0,0)) { TranslateMessage(&msg;); DispatchMessage(&msg;); } return msg.wParam; } // // // //编写窗口过程函数 LRESULT CALLBACK WinProc( HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { int xogr,yogr,a,b,i; float xbegin[3],ybegin[3],xend[3],yend[3]; HDC hDC; PAINTSTRUCT ps; HBRUSH hBrush; HPEN hPen; RECT clientRect; COLORREF color[4]={RGB(255,0,0),RGB(0,255,0),RGB(0,0,255),RGB(255,255,255)}; int BrushStyle[3]={BLACK_BRUSH,GRAY_BRUSH,WHITE_BRUSH}; switch(uMsg) { case WM_PAINT: hDC=BeginPaint(hwnd,&ps;); GetClientRect(hwnd,&clientRect;); xogr=clientRect.left+100; yogr=clientRect.top+100; a=xogr+200; b=yogr+200; xbegin[0]=400; xbegin[1]=400-100*2.732; xbegin[2]=xbegin[1]; ybegin[0]=b-100; ybegin[1]=yogr; ybegin[2]=400; xend[0]=xbegin[1]; xend[1]=xbegin[2]; xend[2]=xbegin[0]; yend[0]=ybegin[1]; yend[1]=ybegin[2]; yend[2]=ybegin[0]; for(i=0;i<3;i++) { SelectObject(hDC,hPen); hBrush=CreateSolidBrush(color[i]); SelectObject(hDC,hBrush); Pie(hDC,xogr,yogr,a,b,xbegin[i],ybegin[i],xend[i],yend[i]); Sleep(100); } InvalidateRect(hwnd,NULL,1); DeleteObject(hPen); DeleteObject(hBrush); EndPaint(hwnd,&ps;); break; case WM_TIMER: if(wParam==1111) InvalidateRect(hwnd,NULL,1); break; case WM_CLOSE: if(IDYES==MessageBox(hwnd,"是否真的结束?","message",MB_YESNO)) { DestroyWindow(hwnd); } break; case WM_DESTROY: PostQuitMessage(0); break; default: return DefWindowProc(hwnd,uMsg,wParam,lParam); break; } return 0; }
实验1 Windows编程基础实验报告 一、实验目的和要求 (1)创建一个Win32应用程序Ex_SDK,在程序中构造一个编辑框控件和一个按钮。 (2)编辑框用于输入一元二次方程的系数,当单击"计算"按钮,获取方程系数得的根 通过TextOut显示在窗口客户区中。 二、实验环境 操作系统:Windows 2000 编译器:Visual C++ 6.0的开发环境 三、实验准备和说明 (1)具备知识:简单的SDK编程基础。 (2)准备本次上机所需要的程序。 (3)创建本次实验工作文件夹"…\Visual C++程序\实验\实验1" 四、实验内容和步骤 1.启动Visual C++ 6.0 打开计算机,启动Visual C++ 6.0系统。 2.创建工程并添加代码 选择"文件" "新建"菜单命令,打开应用程序向导,显示出"新建"对话框。单击"工程"标 签,从列表框中选中Win32 Application(Win32 应用程序)项。在工程名称框中输入Win32应用程序项目名称Ex_SDK。单击浏览按钮... 将工程定位到文件夹 "…\Visual C++ 6.0程序\实验\实验1"。 单击"确定"按钮继续。在向导第一步对话框中,选中An empty project(一个空的工程)项。单击"完成"按钮,系统将显示AppWizard(应用程序向导 )的创建信息,单击"确定"按钮,系统将自动创建此应用程序。 再次选择"文件" "新建"菜单命令,显示出"新建"对话框。单击"文件"标签,在左边的列 表框中选择C++ Source File项,在右边的"文件"下的编辑框中输入Ex_SDK.cpp,单击"确定"按钮。 在打开的文档窗口中输入下面的代码: #include #include #include // 求一元二次方程的根,函数返回根的个数 int GetRoot(float a, float b, float c, double *root) { double delta, deltasqrt; delta = b*b - 4.0 * a * c; if (delta<0.0) return 0; // 无根 deltasqrt = sqrt(delta); if (a!=0.0) { root[0] = (-b + deltasqrt)/(2.0 * a); root[1] = (-b - deltasqrt)/(2.0 * a); } else if (b!=0.0) root[0] = root[1] = -c/b; else return 0; if (root[0] == root[1]) return 1; else return 2; } char str[80]; LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM); // 窗口过程 int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { HWND hwnd ; // 窗口句柄 MSG msg ; // 消息 WNDCLASS wndclass ; // 窗口类 wndclass.style = CS_HREDRAW " CS_VREDRAW ; wndclass.lpfnWndProc = WndProc ; wndclass.cbClsExtra = 0 ; wndclass.cbWndExtra = 0 ; wndclass.hInstance = hInstance ; wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ; wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ; wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ; wndclass.lpszMenuName = NULL ; wndclass.lpszClassName = "SDKWin"; // 窗口类名 if (!RegisterClass (&wndclass)) // 注册窗口 { MessageBox (NULL, "窗口注册失败!", "HelloWin", 0) ; return 0 ; } // 创建窗口 hwnd = CreateWindow ("SDKWin", // 窗口类名 "实验1——Windows编程基础", // 窗口标题 WS_OVERLAPP
      在程序员眼中,Windows的用户界面就是一个由无数个大小窗口组合在一起的整体。密码框也不例外, 它是一个具有ES_PASSWORD风格的"TEdit"类子窗口控制。既然它是一个窗口,就难免具有一些Windows窗口所共有的特性: 有一个窗口过程;可以接收消息。 或许你已经知道,向文本框发送一个WM_GETTEXTLENGTH消息,就能获得文本框中的字符串长度。 如果向文本框发送一个WM_GETTEXT消息,就能获得文本框中的字符串。这两个消息对密码框同样有效,因为它们都是基于 "TEdit"类所创建的子窗口控制,只是风格不同罢了。 我们可以用以下两行代码来获取密码了: iLength = SendMessage(hwnd, WM_GETTEXTLENGTH, 0, 0) ; SendMessage(hwnd, WM_GETTEXT, (WPARAM)(iLength + 1), (LPARAM)(pStrPassWord)) ; 问题1:SendMessage函数需要密码框的窗口句柄作为参数, 如何得到密码框的窗口句柄呢? 很简单,使用WindowFormPoint API函数。该函数接受一个POINT类结构的参数, 并返回包含该点的窗口句柄,函数原型如下: HWND WindowFromPoint(POINT point) ; 知识准备充足,代码如下: /*------------------------------------ *号查看器v1.0 20110407 Ra -------------------------------------*/ #include #define WM_GETCODE WM_USER+1 LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM) ; LRESULT CALLBACK EditProc(HWND, UINT, WPARAM, LPARAM) ; WNDPROC OldProc ; POINT point ; int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpszCmdLine, int iCmdShow) { static TCHAR szAppName[] = TEXT("GetCode") ; MSG msg ; HWND hwnd ; WNDCLASS wndclass ; wndclass.style = CS_HREDRAW | CS_VREDRAW ; wndclass.lpfnWndProc = WndProc ; wndclass.cbClsExtra = 0 ; wndclass.cbWndExtra = 0 ; wndclass.hInstance = hInstance ; wndclass.hIcon = LoadIcon(NULL, IDI_WINLOGO) ; wndclass.hCursor = LoadCursor(NULL, IDC_ARROW) ; wndclass.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1) ; wndclass.lpszMenuName = NULL ; wndclass.lpszClassName = szAppName ; if (!RegisterClass(&wndclass)) { MessageBox(NULL, TEXT("This program requires Windows NT!") , szAppName, MB_ICONERROR) ; return 0 ; } hwnd = CreateWindowEx(WS_EX_TOPMOST, szAppName, TEXT("*号查看器"), WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hInstance, NULL) ; ShowWindow(hwnd, iCmdShow) ; UpdateWindow(hwnd) ; while (GetMessage(&msg, NULL, 0, 0)) { TranslateMessage(&msg) ; DispatchMessage(&msg) ; } return msg.wParam ; } LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { static HWND hwndEdit ; static HINSTANCE hInstance ; switch(message) { case WM_CREATE: hInstance = (HINSTANCE)GetWindowLong(hwnd, GWL_HINSTANCE) ; hwndEdit = CreateWindow(TEXT("button"), TEXT("请将此按钮拖放到你想查看的*号上"), WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON, 0, 0, 0, 0, hwnd, (HMENU)1,hInstance, NULL) ; OldProc = (WNDPROC)SetWindowLong(hwndEdit, GWL_WNDPROC, (LONG)EditProc) ; //改变指定窗口的属性 //GWL_WNDPROC:为窗口过程设定一个新的地址 return 0 ; case WM_SIZE: TEXTMETRIC tm ; HDC hdc ; int cxChar, cyChar, cxScreen, cyScreen, iWndWidth, iWndHeight ; //确定主窗口宽度与高度 cxScreen = GetSystemMetrics(SM_CXSCREEN) ; cyScreen = GetSystemMetrics(SM_CYSCREEN) ; iWndWidth = cxScreen / 10 * 4 ; iWndHeight = cyScreen / 4 ; //确定字符宽度与高度 hdc = GetDC(hwnd) ; GetTextMetrics(hdc, &tm) ; cxChar = tm.tmAveCharWidth ; cyChar = tm.tmHeight + tm.tmExternalLeading ; MoveWindow(hwnd, cxScreen / 100, cyScreen / 100, iWndWidth, iWndHeight, TRUE) ; MoveWindow(hwndEdit, (iWndWidth - cxChar * 38) / 2, iWndHeight / 3, cxChar * 38, cyChar + 10, TRUE) ; //对指定的窗口设置键盘焦点。该窗口必须与调用线程的消息队列相关。 SetFocus(hwnd) ; return 0 ; case WM_GETCODE: HWND hwndDst ; int iLength ; TCHAR PassWord[255], ClassName[255] ; //获得包含指定点的窗口的句柄。 hwndDst = WindowFromPoint(point) ; if (hwndDst == hwndEdit || hwndDst == hwnd) { MessageBox(hwnd, TEXT("如果你有任何意见或建议,请与我联系:td1126@163.com"), TEXT("关于 Ra"), MB_ICONINFORMATION) ; return 0 ; } iLength = GetClassName(hwndDst, ClassName, 255) ; if (!iLength) { MessageBox(hwnd, TEXT("获取类名失败!"), TEXT("Error"), MB_ICONERROR) ; return 0 ; } if (strncmp((char *)ClassName, (char *)"TEdit",4) != 0) { MessageBox(hwnd, TEXT("目标位置不是一个密码框!"), TEXT("Error"), MB_ICONERROR) ; //return 0 ; } iLength = SendMessage(hwndDst, WM_GETTEXTLENGTH, 0, 0) ; SendMessage(hwndDst, WM_GETTEXT, (WPARAM)(iLength + 1), (LPARAM)PassWord) ; MessageBox(hwnd, PassWord, TEXT("你想得到的密码是:"), MB_OK) ; return 0 ; case WM_DESTROY: PostQuitMessage(0) ; return 0 ; } return DefWindowProc(hwnd, message, wParam, lParam) ; } LRESULT CALLBACK EditProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { static HCURSOR hOldCursor, hMyCursor ; switch(message) { case WM_LBUTTONDOWN: //从一个与应用事例相关的可执行文件(EXE文件)中载入指定的光标资源。 hMyCursor = LoadCursor(NULL, MAKEINTRESOURCE(IDC_CROSS)) ; //该函数确定光标的形状。 hOldCursor = SetCursor(hMyCursor) ; SetCapture(hwnd) ; return 0 ; case WM_LBUTTONUP: //从当前线程中的窗口释放鼠标捕获,并恢复通常的鼠标输入处理。捕获鼠标的窗口接收所有的鼠标输入(无论光标的位置在哪里),除非点击鼠标键时,光标热点在另一个线程的窗口中。 ReleaseCapture() ; SetCursor(hOldCursor) ; point.x = LOWORD(lParam) ; point.y = HIWORD(lParam) ; //将指定点,或者矩形的用户坐标转换成屏幕坐标。 ClientToScreen(hwnd, &point) ; SendMessage(GetParent(hwnd), WM_GETCODE, 0, 0) ; return 0 ; } return CallWindowProc(OldProc, hwnd, message, wParam, lParam) ; } WM_GETCODE是用户自定义消息,根据Windows编程规则,它的值应大于0X0400,我们把它定义为WM_USER+1,即:0X0401。point是一个POINT类全局结构变量,用来保存密码框的坐标值。 主窗口过程在WM_CREATE消息期间创建了一个按钮控制hwndEdit,并使用窗口子类化技术给它安装了一个钩子,捕获它的WM_LBUTTONDOWN、WM_LBUTTONUP消息。这时,Windows的内部窗口过程就不能再处理这两个消息,所以,我们在按钮上单击鼠标左键时,不再看到按钮被按下并弹起的视觉效果,但这点对我们并不重要。 当用户在按钮控制上按下鼠标左键时,Windows向按钮控制发送一个WM_LBUTTONDOWN消息,这个消息被我们编写的EditProc窗口过程捕获,我们对这个消息的处理很简单:将鼠标光标设置为十字型,并捕获鼠标。鼠标被捕获后,所有的鼠标消息都将被发送到按钮控制的窗口过程,即EditProc。但我们只处理鼠标消息的WM_LBUTTONUP消息,而使用 return CallWindowProc(OldProc, hwnd, message, wParam, lParam) ; 将其它的消息交由Windows的内部窗口过程处理。 在WM_LBUTTONUP消息期间,我们先撤消对鼠标的捕获,再将鼠标光标恢复原状,然后保存鼠标左键释放时的坐标位置。注意,这个坐标值是相对于按钮控制的客户区坐标,所以我们必须使用ClientToScreen函数将它转换成屏幕坐标。最后,使用SendMessage函数给主窗口发送一个WM_GETCODE自定义消息,把余下的工作交给主窗口过程处理。 主窗口过程的WM_GETCODE消息逻辑,是程序的核心。它先使用WindowFromPoint函数获得鼠标释放时,鼠标光标所在位置的窗口句柄。如果鼠标是在本程序的窗口内释放,将弹出一个关于消息框。否则,就将获得的窗口句柄进行判断,如果是一个密码框,就给它发送WM_GETTEXTLENGTH、WM_GETTEXT消息,获取密码。 问题2: 利用本程序为什么不能查看QQ密码? 如果你确实用本程序查看过QQ密码的话,那么你发现你获取的窗口一直是主窗口(父窗口),因为QQ页面设置时利用了一项新技术,具体叫什么名字我忘记了,QQ页面中并没有其他的控件,帐户、密码的输入框并不是windows控件,而是类似于画上去的图形,所以我们无法获取它的句柄,从而无法查看QQ密码。

16,472

社区成员

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

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

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