Visual Studio 2022写Windows程序造成CPU占用率过高故障排除

微软技术分享 分享大师
全栈领域优质创作者
博客专家认证
2024-02-12 09:48:16

 我是荔园微风,作为一名在IT界整整25年的老兵,今天针对Visual Studio 2022写Windows程序造成CPU占用率过高故障进行排除。

下面是一个标准的Windows程序,也可以说是经典程序了,但是这个程序一运行,WinMain.exe的CPU占用率接近100%,为什么会这样??这个时候一定要冷静,要好好回想windows程序的运行机制原理,知道原理了,吃透原理了,就能理解为什么会有这种故障现象,这对初学者要求较高,但这也是让初学者成长的最好的方法。

#include<windows.h>
#include<stdio.h>
 
LRESULT CALLBACK WinFirstProc(
  HWND hwnd,  // handle to window
  UINT uMsg,  // message identifier
  WPARAM wParam,  //first message parameter
  LPARAM lParam    //second message parameter
);
 
int WINAPI WinMain(
  HINSTANCE hInstance,  //handle to current instance
  HINSTANCE hPrevInstance,  // handle to previous instance
  LPSTR lpCmdLine,  //command line
  int nCmdShow  // show state
)
{
  //设计一个窗口类
  WNDCLASS wndcls;
  wndcls.cbClsExtra=0;
  wndcls.cbWndExtra=0;
  wndcls.hbrBackground=(HBRUSH)GetStockObject(BLACK_BRUSH);
  wndcls.hCursor=LoadCursor(NULL,IDC_CROSS);
  wndcls.hIcon=LoadIcon(NULL, IDI_ERROR);
  wndcls.hInstance=hInstance;  //应用程序实例句柄由WinMain函数传进来
  wndcls.lpfnWndProc=WinFirstProc;
  wndcls.lpszClassName="csdn";
  wndcls.lpszMenuName=NULL;
  wndcls.style=CS_HREDRAW | CS_VREDRAW;
  RegisterClass(&wndcls);
  
   //创建窗口,定义一个变量用来保存成功创建窗口后返回的句柄
   HWND hwnd;
   hwnd=CreateWindow("csdn","www.csdn.net",WS_OVERLAPPEDWINDOW,0,0,600,400,NULL,NULL,hInstance,NULL);
  
   //显示及刷新窗口
  ShowWindow(hwnd,SW_SHOWNORMAL);
  UpdateWindow(hwnd);
 
  //定义消息结构体,开始消息循环
  MSG msg;
  while(GetMessage(&msg,hwnd,0,0))
   {
  TranslateMessage(&msg);
  DispatchMessage(&msg);
   }
  return msg.wParam;
}
 
   //编写窗口过程函数
  LRESULT CALLBACK WinFirstProc(
  HWND hwnd,  // handle to window
  UINT uMsg,  // message identifier
  WPARAM wParam,  // first message parameter
  LPARAM lParam  // second message parameter
)
{
  switch(uMsg)
   {
  case WM_CHAR:
      char szChar[20];
      sprintf_s(szChar, sizeof(szChar), "char is %d", wParam);
      MessageBox(hwnd, szChar, "char", 0);
      break;
  case WM_LBUTTONDOWN:
      MessageBox(hwnd,"mouse clicked","message",0);
      HDC hdc;
      hdc=GetDC(hwnd);  //不能在响应WM_PAINT消息时调用
      TextOut (hdc,0,50,"csdn"strlen("csdn"));
      ReleaseDC(hwnd, hdc);
      break;
  case WM_PAINT:
      HDC hDC;
      PAINTSTRUCT ps;
      hDC=BeginPaint(hwnd,&ps); //BeginPaint只能在响应WM_PAINT消息时调用 
        TextOut(hDC,0,0,"www.csdn.net",strlen("www.csdn.net"));
      EndPaint(hwnd, &ps);
      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);
  }
  return 0;
}

这个故障其实是一个消息循环的错误分析。解决的方法主要是,一方面你可以查CSDN,另一方面,你可以查MSDN,这两个地方查过了,问题基本就可以解决了。

注意下面这些代码中的第三行:

//定义消息结构体,开始消息循环
MSG msg;
while(GetMessage(&msg,hwnd,0,0))
{
   TranslateMessage(&msg);
  DispatchMessage(&msg);
}
return msg.wParam;

这段代码程序只有一个窗口,而我们前面说了 GetMessage函数的hWnd参数用于指定接收属于哪一个窗口的消息,于是上面程序就在消息循环中为 GetMessage函数的 hWnd参数指定了 CreateWindow函数返回的窗口句柄。如果这样,然后运行程序,再关闭程序。你会发现你的计算机变慢了,同时按下键盘上的“Ctrl+Alt+Delete”键,启动Windows的任务管理器,切换到“进程”选项卡,单击“CPU”项进行排序,你会发现WinMain.exe的CPU占用率非常高,那么这是什么原因呢?

在GetMessage函数中,曾说过如果hWnd参数是无效的窗口句柄或 lpMsg 参数是无效的指针,则 GetMessage 函数将返回-1。当我们关闭窗口时,调用了 DestroyWindow 来销毁窗口,由于窗口被销毁了,窗口的句柄当然也就是无效的句柄了,那么GetMessage将返回-1。在C/C++语言中,非0即为真,由于窗口被销毁,句柄变为无效,GetMessage总是返回-1,循环条件总是为真,于是形成了一个死循环,计算机当然就变慢了。

在 MSDN 关于 GetMessage函数的说明文档中给出了下面的代码:

BOOL bRet;

while( (bRet = GetMessage( &msg, NULL, 0, 0 )) != 0)
{
   if (bRet == -1)
    {
      // handle the error and possibly exit
    } 
  else
  {
      TranslateMessage(&msg);
      DispatchMessage(&msg);
  }
}

针对我们这个问题,可以修改上述代码如下:

HWND hwnd;
hwnd=CreateWindow(..);

…

MSG msg;
BOOL bRet;

while( (bRet = GetMessage( &msg, hwnd, 0, 0 )) != 0)
{
  if (bRet == -1)
  {
      // handle the error and possibly exit
      return -1;
   }
  else
  {
      TranslateMessage(&msg);
      DispatchMessage(&msg);
  }
}

这样就可以排除这个故障了。

类似的各类故障还有很多很多,作为windows程序的初学者,一定不要怕这些故障,要多查CSDN,如果CSDN上没有相关的帖子,那就查MSDN,MSDN上也许不能像CSDN上这样直接给出答案,但是能让你从根本上来理解问题可能存在的地方。如果这样,我们就能通过解决一次故障而更深入的理解内部程序运行的机制,进而真正提高自己的真实水平。而这种水平的提升,是你在任何书本中都学不到的。根据我的经验,一个初学者如能用这个方法解决60至80个这样的同类问题,就能深入windows程序调试的理解,真正向高手前进了。

各位小伙伴,下次我们再深入研究windows程序的运行机制。


文章来源: https://blog.csdn.net/wang2015cn/article/details/131351715
版权声明: 本文为博主原创文章,遵循CC 4.0 BY-SA 知识共享协议,转载请附上原文出处链接和本声明。


...全文
17 回复 打赏 收藏 转发到动态 举报
写回复
用AI写文章
回复
切换为时间正序
请发表友善的回复…
发表回复

395

社区成员

发帖
与我相关
我的任务
社区描述
微软技术社区为中国的开发者们提供一个技术干货传播平台,传递微软全球的技术和产品最新动态,分享各大技术方向的学习资源,同时也涵盖针对不同行业和场景的实践案例,希望可以全方位地帮助你获取更多知识和技能。
windowsmicrosoft 企业社区
社区管理员
  • 微软技术分享
  • 郑子铭
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

微软技术社区为中国的开发者们提供一个技术干货传播平台,传递微软全球的技术和产品最新动态,分享各大技术方向的学习资源,同时也涵盖针对不同行业和场景的实践案例,希望可以全方位地帮助你获取更多知识和技能。

予力众生,成就不凡!微软致力于用技术改变世界,助力企业实现数字化转型。

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