用C语言和EasyX库写一个五子棋,我踩过的这些坑你别再踩了
用C语言和EasyX库写五子棋那些教科书不会告诉你的实战陷阱第一次用EasyX库写五子棋时我以为三天就能搞定结果花了三周时间调试各种奇葩问题。坐标计算差1个像素导致棋子永远对不齐、鼠标点击识别区域偏差、二维数组越界导致程序崩溃...这些坑每一个都能让新手抓狂。下面分享的不仅是解决方案更是一套图形界面编程的调试方法论。1. 图形界面初始化从第一个像素开始就埋雷EasyX的initgraph(600,600)看似简单但实际开发中至少有三个隐藏陷阱等着你// 典型错误示例直接照搬模板代码 initgraph(600, 600); loadimage(NULL, _T(background.png));致命问题1路径黑洞当你的图片加载失败时程序不会报错而是静默运行。去年有个学生问我为什么棋盘显示不出来最后发现是VS默认的工作目录在项目文件夹外的x64/Debug而他的图片放在项目根目录。两种解决方案// 方案A使用绝对路径不推荐 loadimage(NULL, _T(D:\\project\\gomoku\\background.png)); // 方案B修改VS配置 // 项目属性 → 调试 → 工作目录 → 改为$(ProjectDir)致命问题2资源释放很多教程不会告诉你当窗口意外关闭时比如用户点X如果没有处理WM_CLOSE消息会导致内存泄漏。正确的初始化应该这样写if (initgraph(600, 600) ! 0) { MessageBox(GetHWnd(), _T(图形窗口初始化失败), _T(错误), MB_OK); return -1; } // 设置关闭按钮处理 SetWindowText(GetHWnd(), _T(五子棋 (点击右上角X可退出)));致命问题3DPI缩放在高分辨率屏幕上比如4K显示器你的600x600窗口可能实际只占屏幕一角。这时鼠标坐标会按缩放比例偏移导致点击位置不准。解决方案// 在initgraph后立即添加 SetProcessDPIAware(); // 告诉系统不要缩放我们的窗口2. 棋盘绘制你以为的直线根本不是直线用line()函数画棋盘时这个看似简单的操作有至少两个大坑// 常见错误画法 setlinecolor(BLACK); for (int x 0; x 600; x 40) line(x, 0, x, 600); // 竖线 for (int y 0; y 600; y 40) line(0, y, 600, y); // 横线坑1边缘消失当线条紧贴窗口边缘时在某些显卡驱动下会被裁剪。解决方案是留出1像素边距// 正确画法边缘留白 const int MARGIN 1; // 1像素边距 for (int x 20; x 580; x 40) line(x, 20, x, 580); for (int y 20; y 580; y 40) line(20, y, 580, y);坑2线宽玄学默认线宽是1像素但在某些显示器上会显得模糊。可以通过setlinestyle增强显示效果// 设置抗锯齿线条 setlinestyle(PS_SOLID | PS_ENDCAP_SQUARE, 2); // 2像素宽3. 鼠标交互点击误差的终极解决方案获取鼠标消息看似简单但实际开发中会遇到三个典型问题ExMessage m; while (true) { m getmessage(EX_MOUSE); if (m.message WM_LBUTTONDOWN) { // 下棋逻辑... } }问题1点击灵敏度用户快速双击时可能误触发多次落子。解决方案是增加点击间隔判断static DWORD lastClickTime 0; DWORD currentTime GetTickCount(); if (currentTime - lastClickTime 300) // 300毫秒内不重复处理 continue; lastClickTime currentTime;问题2坐标转换误差将鼠标坐标转换为棋盘索引时直接整除会导致边缘点击不准// 错误转换方式 int i m.x / 40; // 当m.x39时i0m.x40时i1应该使用四舍五入算法// 正确转换方式 int i (m.x 20) / 40; // 20实现四舍五入 if (i 0) i 0; if (i NUM) i NUM - 1; // 边界检查问题3消息队列阻塞在胜负判定后如果直接break会导致窗口无法响应关闭事件。应该改用状态机模式enum GameState { RUNNING, BLACK_WIN, WHITE_WIN } state RUNNING; while (state RUNNING) { if (MouseHit()) { // 非阻塞检查 m getmessage(EX_MOUSE); // 处理鼠标... } if (state ! RUNNING) { // 显示胜利信息... while (!_kbhit()) Sleep(100); // 等待按键 break; } }4. 胜负判定从O(n²)到O(1)的优化之路教科书式的五子棋胜负判定通常是遍历整个棋盘这种O(n²)算法在15x15棋盘上效率尚可但有更聪明的做法优化1增量检查法只需检查最后落子位置的四个方向int CheckDirection(int x, int y, int dx, int dy) { int count 1; // 当前棋子已算1个 for (int i 1; i 5; i) { if (xi*dx NUM || yi*dy NUM || pieceArr[xi*dx][yi*dy] ! pieceArr[x][y]) break; count; } for (int i 1; i 5; i) { if (x-i*dx 0 || y-i*dy 0 || pieceArr[x-i*dx][y-i*dy] ! pieceArr[x][y]) break; count; } return count 5; }优化2位运算加速对于高级开发者可以用位棋盘表示状态用位运算并行检测// 黑棋和白棋各用一个unsigned int[15]表示 unsigned int blackRows[NUM] {0}; unsigned int whiteRows[NUM] {0}; // 落子时更新 if (isBlack) blackRows[y] | 1 x; else whiteRows[y] | 1 x; // 检测胜利用预计算的掩码做位与操作 const unsigned int winMasks[] { 0x1F, 0x3E, 0x7C, ..., // 水平5连 0x108421, 0x210842, ... // 垂直5连 };5. 那些让程序崩溃的内存问题二维数组pieceArr[NUM][NUM]是五子棋的核心数据结构但也是崩溃的重灾区陷阱1数组越界当鼠标点击窗口边缘时转换的数组索引可能越界// 错误代码 int i m.x / 40; int j m.y / 40; if (pieceArr[i][j] ! 0) // 当i/j15时越界 return false;解决方案边界检查应该先检查索引有效性bool IsValidPosition(int x, int y) { if (x 0 || x NUM || y 0 || y NUM) return false; return pieceArr[x][y] 0; }陷阱2栈溢出当NUM过大时比如100x100局部数组可能导致栈溢出。改为动态分配int (*pieceArr)[NUM] malloc(NUM * sizeof(*pieceArr)); if (!pieceArr) { // 处理内存不足 } // 使用后记得free6. EasyX的隐藏功能让界面更专业大多数教程只教基础绘图其实EasyX还有这些实用功能功能1透明文字背景默认outtextxy的文字背景不透明会覆盖棋盘线// 设置文字背景模式 setbkmode(TRANSPARENT); settextcolor(RED); outtextxy(250, 0, _T(黑棋胜利));功能2抗锯齿绘图启用高级图形模式可以获得更平滑的绘制效果// 在initgraph后调用 SetWorkingImage(NULL); // 设置操作目标为窗口 BeginBatchDraw(); // 开始批量绘制 // ...所有绘图操作 FlushBatchDraw(); // 批量提交功能3自定义光标隐藏系统光标绘制更美观的游戏光标// 隐藏系统光标 HCURSOR hCur LoadCursor(NULL, IDC_ARROW); SetClassLongPtr(GetHWnd(), GCLP_HCURSOR, (LONG_PTR)hCur); // 在绘制循环中绘制自定义光标 if (state RUNNING) { setfillcolor(black ? RGB(0,0,0) : RGB(255,255,255)); fillcircle(m.x, m.y, 5); // 跟随鼠标的半透明预览棋子 }7. 从项目到产品那些教科书不会教的工程实践完成基本功能后还需要考虑这些生产级问题实践1保存游戏状态添加存档/读档功能需要使用文件操作void SaveGame() { FILE* fp fopen(save.dat, wb); if (fp) { fwrite(pieceArr, sizeof(int), NUM*NUM, fp); fwrite(black, sizeof(bool), 1, fp); fclose(fp); } }实践2添加AI对手最简单的AI可以实现随机落子void AIPlay() { int empty[NUM*NUM][2], count 0; for (int i 0; i NUM; i) for (int j 0; j NUM; j) if (pieceArr[i][j] 0) { empty[count][0] i; empty[count][1] j; count; } if (count 0) { int idx rand() % count; DrawPiece(false, empty[idx][0]*4020, empty[idx][1]*4020); } }实践3网络对战基于TCP协议实现简易网络对战// 服务端 SOCKET servSock socket(AF_INET, SOCK_STREAM, 0); sockaddr_in servAddr { AF_INET, htons(12345) }; bind(servSock, (SOCKADDR*)servAddr, sizeof(servAddr)); listen(servSock, 1); // 客户端 SOCKET clntSock socket(AF_INET, SOCK_STREAM, 0); sockaddr_in clntAddr { AF_INET, htons(12345) }; inet_pton(AF_INET, 127.0.0.1, clntAddr.sin_addr); connect(clntSock, (SOCKADDR*)clntAddr, sizeof(clntAddr));写完这个五子棋项目后最大的体会是图形界面编程就像在雷区跳舞每走一步都可能触发意想不到的问题。那些看似简单的API背后藏着无数细节陷阱。记得第一次看到棋子显示错位时我花了整整两天才发现是DPI缩放问题。而鼠标点击不准的bug最终发现是坐标转换时没考虑四舍五入。这些经验让我明白好的程序员不仅要会写代码更要学会系统性调试——从图形渲染管线到消息处理机制从内存布局到CPU缓存行为每个层面都可能成为问题的根源。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2497773.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!