winapi – Win32 – 什么可能导致不正确的绘图?

我正在创建一个游戏,其中在WM_PAINT消息期间完成了大量绘图.窗口失效有几个不同的地方,迫使它重绘.我把所有东西画到了一个屏幕外的DC,然后把它绘制到窗口 – 创建非闪烁的“帧”.

然而,每隔一段时间,一切都突然开始被错误地绘制.在我使用的五个位图中,前三个正在或多或少(但不完全)正确绘制.同样,所有颜色信息对于这三个都是正确的.在这三个之后绘制的另外两个是用错误的颜色绘制的 – 我认为白色仍然是白色的,但其他一切都被绘制为灰色.我不是说灰度,我的意思是除了白色之外的所有东西都是相同的颜色 – 灰色.

此外,当这种情况开始发生时,通常情况下所有内容都被绘制得太高 – 大约20到30个像素.此外,字体和消息框停止工作 – 所有文本都以默认字体绘制(但奇怪的是,正确的颜色),消息框只出现短暂,没有文字,然后消失 – 但必须像常规一样被解雇(你必须按回车键,否则如果你点击主窗口,它会按照正常情况执行,当有一个消息框打开时输入 – 闪烁并拨打错误音.所以一切都被皇家搞砸了.

我一直在研究这个项目已经有一段时间了,我最近才开始看到这个错误 – 尽管我根本没有修改下面的代码片段.很难测试可能会做什么,因为它似乎偶尔会发生.

这是WndProc中WM_CREATE和WM_PAINT中的代码:

case WM_CREATE:
    {
        hdc = GetDC(hWnd);
        hdcmem = CreateCompatibleDC(hdc);
        RECT rc;
        GetClientRect(hWnd,&rc);
        hdcbm = CreateCompatibleBitmap(hdc,rc.right,rc.bottom);
        hbcmem = CreateCompatibleDC(hdcmem);
        hdcbmold = (HBITMAP)SelectObject(hdcmem,hdcbm);

        // Load bitmaps
        bg = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_BACKGROUND));
        mainCont = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_GAME_CONT));
        if(bg == NULL || mainCont == NULL)
            ThrowError("A bitmap failed to load.");
    }
    break;
case WM_PAINT:
{
    PAINTSTRUCT ps;
    BeginPaint(hWnd,&ps);

    // Background
    hdcold = (HBITMAP)SelectObject(hbcmem,bg);
    BitBlt(hdcmem,237,196,hbcmem,SRCCOPY);
    BitBlt(hdcmem,237 * 2,SRCCOPY);

    // Main Game Container
    hdcold = (HBITMAP)SelectObject(hbcmem,mainCont);
    BitBlt(hdcmem,26,300,SRCCOPY);

    // Side Info
    HBITMAP side = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_SIDEINFO));
    hdcold = (HBITMAP)SelectObject(hbcmem,side);
    BitBlt(hdcmem,339,154,SRCCOPY);
    DrawLevelNumber(game.map.levelnumber);

    if (color)
        sprites = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_COLOR_SPRITES));
    else sprites = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_BLACKWHITE_SPRITES));
    hdcold = (HBITMAP)SelectObject(hbcmem,sprites);

    // Find x and y coordinate for the top left of the visible screen
    int x = game.player.x,y = game.player.y,ypos = 0;
    if (x < 4)  x = 4;
    if (x > 27) x = 27;
    if (y < 4)  y = 4;
    if (y > 27) y = 27;
    x -= 4;
    y -= 4;

    // Draw lower layer
    for (int i = 0; i < 9; i++)
    {
        for (int j = 0; j < 9; j++)
        {
            if (game.map.Layer_Two[x + i][y + j] != 0)
            {
                int xpos = game.get_pos(game.map.Layer_Two[x + i][y + j].get(),ypos,false);
                BitBlt(hdcmem,(i * 32) + 32,(j * 32) + 32,32,xpos,SRCCOPY);
            }
        }
    }

    // Draw upper layer
    for (int i = 0; i < 9; i++)
    {
        for (int j = 0; j < 9; j++)
        {
            if ((game.map.Layer_Two[x + i][y + j] != 0 && game.map.Layer_One[x + i][y + j] >= 64 && game.map.Layer_One[x + i][y + j] <= 111))
            {
                int xpos = game.get_pos(game.map.Layer_One[x + i][y + j].get(),true);
                BitBlt(hdcmem,xpos + 96,SRCPAINT);
                BitBlt(hdcmem,SRCAND);
            } else {
                int xpos = game.get_pos(game.map.Layer_One[x + i][y + j].get(),SRCCOPY);
            }
        }
    }

    // If it isn't started,show title
    if (!game.started)
    {

        HDC tmphdc = CreateCompatibleDC(hdcmem);
        HDC tmp = CreateCompatibleDC(tmphdc);
        RECT rc;
        GetClientRect(hWnd,&rc);
        string str = game.map.leveltitle.substr(0,game.map.leveltitle.length() - 1);
        TCHAR* tch = new TCHAR[str.length()];
        mbstowcs_s(NULL,tch,_tcslen(tch),str.c_str(),str.length());
        HFONT font = CreateFont(25,FW_BOLD,false,ANSI_CHARSET,OUT_DEFAULT_PRECIS,CLIP_DEFAULT_PRECIS,DEFAULT_QUALITY,FF_DONTCARE,NULL);
        SelectObject(tmp,font);
        DrawText(tmp,str.length(),&rc,DT_CALCRECT);
        rc.right += 16;
        HBITMAP tmpbm = CreateCompatibleBitmap(hdcmem,rc.bottom);
        HBITMAP tmpold = (HBITMAP)SelectObject(tmphdc,tmpbm);

        HBRUSH hbr = CreateSolidBrush(RGB(255,255,255));
        HPEN pen = CreatePen(PS_SOLID,1,RGB(255,255));
        SelectObject(hdcmem,pen);
        SelectObject(hdcmem,hbr);
        Rectangle(hdcmem,176 - (rc.right / 2),243,177 + (rc.right / 2),248);
        hbr = CreateSolidBrush(RGB(128,128,128));
        pen = CreatePen(PS_SOLID,RGB(128,128));
        SelectObject(hdcmem,294,299);

        HBITMAP left = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_LEFT));
        hdcold = (HBITMAP)SelectObject(hbcmem,left);
        BitBlt(hdcmem,176 - (rc.right / 2) - 4,4,56,SRCCOPY);
        HBITMAP right = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_RIGHT));
        hdcold = (HBITMAP)SelectObject(hbcmem,right);
        BitBlt(hdcmem,176 + (rc.right / 2) + ((rc.right % 2) != 0),SRCCOPY);

        SelectObject(tmphdc,font);
        SetTextColor(tmphdc,0));
        SetBkColor(tmphdc,RGB(0,0));
        DrawText(tmphdc,DT_CENTER);
        BITMAP structBitmapHeader;
        memset( &structBitmapHeader,sizeof(BITMAP) );
        HGDIOBJ hBitmap = GetCurrentObject(tmphdc,OBJ_BITMAP);
        GetObject(hBitmap,sizeof(BITMAP),&structBitmapHeader);
        BitBlt(hdcmem,247,structBitmapHeader.bmWidth,structBitmapHeader.bmHeight,tmphdc,SRCCOPY);

        DeleteDC(tmphdc);
        DeleteDC(tmp);
    }

    // If paused
    if (game.paused)
    {
        RECT rc;
        rc.top = 32;
        rc.left = 32;
        rc.bottom = 330;
        rc.right = 330;
        BitBlt(hdcmem,288,NULL,BLACKNESS);
        HFONT font = CreateFont(50,FW_NORMAL,NULL);
        SelectObject(hdcmem,font);
        SetTextColor(hdcmem,0));
        SetBkColor(hdcmem,0));
        DrawText(hdcmem,L"PAUSED",6,(DT_CENTER + DT_SINGLELINE + DT_VCENTER));
    }

    nums = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_NUMBERS));
    hdcold = (HBITMAP)SelectObject(hbcmem,nums);
    for (int i = 100; i > 0; i /= 10) // coins and time left
    {
        int tmp;
        if (i == 100)
            tmp = game.coinsleft / 100;
        if (i == 10)
            tmp = ((game.coinsleft % 100) - (game.coinsleft % 10)) / 10;
        if (i == 1)
            tmp = game.coinsleft % 10;
        if (game.coinsleft < i && i > 1)
            tmp = 10;
        int ypos = game.get_num_pos(tmp,(game.coinsleft == 0));
        BitBlt(hdcmem,417 + ((3 - (int)floor(log10((double)i)) * 17)),215,17,23,SRCCOPY);

        if (i == 100)
            tmp = game.timeleft / 100;
        if (i == 10)
            tmp = ((game.timeleft % 100) - (game.timeleft % 10)) / 10;
        if (i == 1)
            tmp = game.timeleft % 10;
        if (game.timeleft < i && i > 1)
            tmp = 10;
        if (game.map.timelimit == 0)
            tmp = 11;
        ypos = game.get_num_pos(tmp,(game.timeleft < 16 || game.map.timelimit == 0));
        BitBlt(hdcmem,369 + ((3 - (int)floor(log10((double)i))) * 17),125,SRCCOPY);
    }

    if (game.onhint)
    {
        HBITMAP sidebg = LoadBitmap(hInst,MAKEINTRESOURCE(IDB_SIDEBG));
        hdcold = (HBITMAP)SelectObject(hbcmem,sidebg);
        BitBlt(hdcmem,353,165,127,146,SRCCOPY);
        HFONT font = CreateFont(18,true,255));
        SetBkColor(hdcmem,0));
        RECT rc;
        rc.top = 168;
        rc.left = 356;
        rc.bottom = 308;
        rc.right = 477;
        string str = "Hint: " + game.map.hint;
        TCHAR* tch = new TCHAR[str.length()];
        mbstowcs_s(NULL,str.length());
        DrawText(hdcmem,DT_CENTER + DT_WORDBREAK);
    } else {
        hdcold = (HBITMAP)SelectObject(hbcmem,sprites);                // LOWER SIDE INFO
        if (game.player.key1 > 0)
            BitBlt(hdcmem,352,192,160,SRCCOPY);
        else BitBlt(hdcmem,SRCCOPY);
        if (game.player.key2 > 0)
            BitBlt(hdcmem,384,SRCCOPY);
        if (game.player.key3 > 0)
            BitBlt(hdcmem,416,224,SRCCOPY);
        if (game.player.key4)
            BitBlt(hdcmem,448,SRCCOPY);
        if (game.player.mod1)
            BitBlt(hdcmem,279,256,SRCCOPY);
        if (game.player.mod2)
            BitBlt(hdcmem,SRCCOPY);
        if (game.player.mod3)
            BitBlt(hdcmem,320,SRCCOPY);
        if (game.player.mod4)
            BitBlt(hdcmem,SRCCOPY);
    }
    BitBlt(hdc,518,401,hdcmem,SRCCOPY);

    EndPaint(hWnd,&ps);
}
    break;
case WM_DESTROY:
    SelectObject(hdc,hdcold);
    DeleteDC(hdcmem);
    DeleteDC(hbcmem);
    ReleaseDC(hWnd,hdc);
    DeleteObject(bg);
    PostQuitMessage(0);
    break;

我也非常感谢任何人可以提供的关于语法,效率和/或更好的代码处理方式的建议/批评.我是Win32的新手.

解决方法

您正在泄漏GDI对象.

每次创建画笔或加载位图时,都会创建一个GDI对象. Windows限制了您可以创建的GDI对象的数量.如果您反复创建对象而不删除它们,您将达到限制并且其他对象的创建将失败.当发生这种情况时,您的显示器看起来会出错 – 您会看到错误的颜色,默认字体等.

任务管理器将显示进程已分配的GDI对象的数量(在“进程”选项卡中,转到“查看”|“选择列”).价值可能会有一点波动,但不应该随着时间的推移而增长.

使用GDI的标准模式是:

>创建对象(例如,使用CreateSolidBrush的画笔).
>使用SelectObject将对象选择到设备上下文中,并记住原始对象(返回值).
>做你的绘画或其他什么.
>使用SelectObject将原始对象选回设备上下文. (当然,这也会从设备上下文中取消选择您的对象.)
>使用DeleteObject删除对象.

您只执行第2步的一半并跳过第4步和第5步.

然后请注意,在每次喷涂操作中都不需要重复步骤1和5.您可以分别将它们移动到程序初始化和终止.并且可以跳过程序终止时的清理,因为OS无论如何都要这样做.

最后,加载位图是一个适度缓慢的操作,所以你绝对不想重复这样做.这是GDI提供对象句柄的要点.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


文章浏览阅读2.2k次,点赞6次,收藏20次。在我们平时办公工作中,很多时候我们经常会使用到虚拟机来进行环境的测试,我们平时在虚拟机上接触的最多的莫过于Linux和Winwdos。不过虚拟机环境和物理机环境是无法直接传输的,那么有的时候呢,同学们又想要在两者之间相互传输文件,可能就会使用QQ邮箱等形式来传输,这样的效率又慢而且繁琐,今天我就为大家带来一种非常便捷的传输方式。通过XFTP工具来进行文件传输。_xftp连接windows
文章浏览阅读1k次。解决 Windows make command not found 和 安装 GCC 环境_windows下载gcc
文章浏览阅读3.2k次,点赞2次,收藏6次。2、鼠标依次点击“计算机配置“ - ”管理模板“ - ”网络“ - ”Lanman工作站”,点击右侧的“启用不安全的来宾登录”策略。Windows访问samba共享时,提示“你不能访问此共享文件夹,因为你组织的安全策略阻止未经身份验证的来宾访问”1、键盘按下window+R键,输入gpedit.msc,启动本地组策略编辑器。首先在终端中输入sudo ufw status查看当前防火墙状态。默认状态是“未配置”,修改为“已启用”。示例:创建一个narada的目录在/home下。1.更新apt储存库列表。_ubuntu samba 目标文件夹访问被拒绝
文章浏览阅读1.3w次。蓝光版属于高清版的一种。BD英文全名是Blu-ray Disc,一种高清的电影版本,这种电影十分清晰但是数据量巨大,占数十G甚至上百G的容量,只有蓝光光碟才能装得下,所以这种高清电影被称为BD版。一般的高清电影多半是从蓝光电影、国外的高清电视频道上压制而来的,可以通过网络下载,多数都经过二次压缩,画质要逊于原视频,不过压缩后的容量从蓝光的25G-50G会减少成4G-8G等(15G-20G不等)。众所周知,视频有两种常见的清晰度,BD和HD,在看电影的时候最常出现这两个标志,那么BD和HD具体指的是什么呢?_bd hd
文章浏览阅读974次,点赞7次,收藏8次。提供了更强大的功能,因为它允许直接访问当前元素,而不需要类型转换。接口,它可以提供一个迭代器,用于按顺序访问集合中的元素。接口是只读的,它只能支持前向迭代,不能修改集合中的元素。类型的集合实例,并向其中添加了几个元素。接口,可以创建一个能够迭代访问泛型集合中元素的迭代器。接口,我们可以在 C# 中实现可迭代的集合,并使用。循环和迭代器手动遍历集合,并输出每个元素的值。接口表示一个可枚举的集合,它定义了一个方法。属性,用于获取集合中当前位置的元素。存储集合中的元素,并实现了。的泛型集合类,它实现了。
文章浏览阅读1.4w次,点赞5次,收藏22次。如果使用iterator的remove方法则会正常,因为iterator的remove方法会在内部调用List的remove方法,但是会修改excepedModCount的值,因此会正常运行。因为遍历过程中进行remove 操作时,该位置后面的元素会挤到前面来,这时候会发生一种情况就是原来元素的位置会被他后面的元素取代,而该位置已经遍历过了,所以该元素不会背遍历。当我们倒序遍历元素的时候,无论删除元素之后的元素怎么移动,之前的元素对应的索引(index)是不会发生变化的,所以在删除元素的时候不会发生问题。_list删除某个元素
文章浏览阅读2.9w次,点赞45次,收藏192次。Windows下配置Visual Studio _vs2022环境变量配置
文章浏览阅读7w次,点赞162次,收藏778次。pip 是Python包管理工具,提供了对 Python 包的查找、下载、安装、卸载的功能,目前Python 3.4 和 2.7 及以上版本都有配套安装,一般pip的位置在...pythonScripts文件夹里面,而在其他版本需要自行下载。_python pip install安装
文章浏览阅读5.8k次,点赞2次,收藏12次。①此电脑右击----->选择属性----->高级系统设置----->环境变量----->path----->编辑----->新建。第一个选项意思就是将安装路径填入到系统环境变量中,这里勾选,后面使用可能会出现问题,建议不要勾选,安装好之后手动添加环境变量。注意:如果提示conda不是内部或外部命令,原因是Anaconda的环境变量没配置好。如果不想立即打开anaconda,不勾选直接finish就好。②输入 conda --version ,查看conda环境。②直接按win键,搜索“环境变量”_windows安装anaconda
文章浏览阅读5.1k次,点赞8次,收藏55次。Windows 系统从零配置 Python 环境,安装CUDA、CUDNN、PyTorch 详细教程_windows cuda cudnn配置
文章浏览阅读1.5w次,点赞54次,收藏68次。macOS系统自带有VNC远程桌面,我们可以在控制端上安装配置VNC客户端,以此来实现远程控制macOS。但通常需要在不同网络下进行远程控制,为此,我们可以在macOS被控端上使用cpolar做内网穿透,映射VNC默认端口5900,通过所生成的公网地址,来实现在公网环境下远程控制VNC。_vnc mac
文章浏览阅读2.4k次,点赞5次,收藏11次。进入后根据自己的电脑系统下载,这是python 3.10版本下载地址,如果想要下载其它版本可进入此链接(下载完成后点击进行安装点击下一步,到这一步时,可以选择将Anaconda添加我的PATH环境变量中,这样就不用自己手动配置和环境变量。安装完成后,打开终端,输出 python 命令可查看是否安装成功。如果显示自己刚才安装的版本号说明安装成功。查看conda版本命令:conda info。_paddlespeech下载
文章浏览阅读3.3k次。所以如果要删除之前新增的课程编译原理,只需输入命令del Course:8:Cname,同时还应该把本课程的学分删除del Course:8:Ccredit,如下图所示;Redis并没有修改数据的命令,所以如果在Redis中要修改一条数据,只能在使用set命令时,使用同样的键值,然后用新的value值来覆盖旧的数据。先调用get命令,输出原先的值,然后set新的值,最后再get得到新值,所以修改成功。输入命令后没有报错,表示成功了,刷新windows的服务,多了一个redis服务。_redis windows服务
文章浏览阅读2.1w次,点赞9次,收藏56次。​​接着在【工作负荷】中,选择【使用C++桌面开发】 ,右边【安装详细信息】去除其它可选项,只勾选【MSVCv142 】和 【Windows 10 SDK】,按图示修改,然后右下角点击安装,之后会有提示让你重启电脑。重启电脑之后,再进行pip安装。报错原因是pip所安装的包需要使用C++编译后才能够正常安装,但是当前安装环境中缺少完整的C++编译环境,因此安装失败。3.安装Microsoft Visual C++ Build Tool离线安装包(1个多G),CSDN资源很多,需要积分下载,_error: microsoft visual c++ 14.0 or greater is required. get it with "micros
文章浏览阅读1.1w次,点赞3次,收藏7次。Step 3: 在右侧窗口中找到名称为“LongPathsEnabled”的“DWORD (32 位) 值”条目,并双击它。通过注册表方法或组策略方法启用长路径支持后,您将能够在 Windows 中使用长路径,并能够访问和处理长路径下的文件和文件夹。Step 2: 依次选择“计算机配置” > “管理模板” > “系统” > “文件资源管理器”。Step 3: 找到“启用 Win32 长路径”设置,双击它。Step 4: 选择“已启用”选项按钮,然后选择“应用”按钮。_windows长路径支持
文章浏览阅读2.5k次,点赞81次,收藏86次。
文章浏览阅读1.3k次,点赞65次,收藏50次。顺序表,链表,栈,队列,ArrayList,LinkedList,Stack,Queue
文章浏览阅读2.3k次,点赞2次,收藏2次。AnyTXTSearcher是一款能够帮助我们对文档以及文本内容进行快速搜索和管理的工具,通过该软件能够搜索各种Office文档,文本文件,代码,PDF文档等,顶级的全文搜索引擎1秒钟之内即可完成搜索。_anytxt searcher
文章浏览阅读8.8k次,点赞73次,收藏70次。有时,在删除/移动/重命名文件夹/文件时,会遇到如下警告,即使将打开的程序关闭了,后台也可能会有没关干净的相关进程。_解除占用
文章浏览阅读4.3w次,点赞91次,收藏102次。JDK(Java Development Kit)是Java开发工具包的缩写,包含了Java编译器、Java虚拟机、Java类库等众多组件,是Java开发的基石,提供了编写、编译和运行Java程序所必需的工具。同时,为了让系统能够正确识别Java环境,在开始使用JDK进行Java开发之前,需要先把JDK安装到本地计算机,并配置好相应的环境变量。本文将介绍JDK安装与环境变量配置的方法。_windows安装jdk并配置环境变量