• 项目实践---贪吃蛇小游戏(下)


    • 对于贪吃蛇小游戏,最主要的还是主函数部分,这里就和大家一一列举出来,
    • 上一章已经写过头文件了,这里就不多介绍了。

    • 首先就是打印桌面,也就是背景,则对应的代码为:
    1. void SetPos(short x, short y)
    2. {
    3. //获取标准输出设备的句柄
    4. HANDLE houtput = NULL;
    5. houtput = GetStdHandle(STD_OUTPUT_HANDLE);
    6. //定位光标的位置
    7. COORD pos = { x,y };
    8. SetConsoleCursorPosition(houtput, pos);
    9. }
    10. void WelcomeToGame()
    11. {
    12. system("cls");
    13. SetPos(40, 14);
    14. wprintf(L"欢迎来到贪吃蛇小游戏\n");
    15. SetPos(42, 20);
    16. system("pause");
    17. system("cls");
    18. SetPos(35, 14);
    19. wprintf(L"用 w.s.a.d来控制蛇的移动,按q加速,e减速\n");
    20. SetPos(35, 15);
    21. wprintf(L"加速能够得到更高的分数\n");
    22. SetPos(42, 20);
    23. system("pause");
    24. system("cls");
    25. }

    • 然后就是打印墙体,在打印墙体之前,要先声明一下类型,用#define,来定义。则对应的代码为:
    1. void CreateMap()
    2. {
    3. //上
    4. int i = 0;
    5. SetPos(0, 0);
    6. for (i = 0; i < 58; i ++)
    7. {
    8. wprintf(L"%lc", WALL);
    9. }
    10. //下
    11. SetPos(0, 26);
    12. for (i = 0; i < 58; i++)
    13. {
    14. wprintf(L"%lc", WALL);
    15. }
    16. //左
    17. for (i = 1; i <= 26; i++)
    18. {
    19. SetPos(0, i);
    20. wprintf(L"%lc", WALL);
    21. }
    22. //右
    23. for (i = 1; i <= 26; i++)
    24. {
    25. SetPos(56, i);
    26. wprintf(L"%lc",WALL);
    27. }
    28. }

    • 然后就是创建蛇身(用链表来实现)和食物,则代码为:
      1. void InitSnake(pSnake ps)
      2. {
      3. int i = 0;
      4. pSnakeNode cur = NULL;
      5. for (i = 0; i < 6; i++)
      6. {
      7. cur = (pSnakeNode)malloc(sizeof(SnakeNode));
      8. if (cur == NULL)
      9. {
      10. perror("InitSnake()::malloc");
      11. return;
      12. }
      13. cur->next = NULL;
      14. cur->x = POS_X + 2 * i;
      15. cur->y = POS_Y;
      16. //头插法插入链表
      17. if (ps->_pSnake == NULL) //空链表
      18. {
      19. ps->_pSnake = cur;
      20. }
      21. else //非空
      22. {
      23. cur->next = ps->_pSnake;
      24. ps->_pSnake = cur;
      25. }
      26. }
      27. cur = ps->_pSnake;
      28. while (cur)
      29. {
      30. SetPos(cur->x, cur->y);
      31. wprintf(L"%lc", BODY);
      32. cur = cur->next;
      33. }
      34. //设置贪吃蛇得属性
      35. ps->_dir = RIGHT; //默认向右
      36. ps->_score = 0;
      37. ps->_food_weight = 10;
      38. ps->_sleep_time = 200; //单位是ms
      39. ps->_status = OK;
      40. }
      41. void CreateFood(pSnake ps)
      42. {
      43. int x = 0;
      44. int y = 0;
      45. //生成2的倍数
      46. //x:2~53
      47. //y:1~25
      48. again:
      49. do
      50. {
      51. x = rand() % 53 + 2;
      52. y = rand() % 25 + 1;
      53. } while (x % 2 != 0);
      54. //x和y的坐标不能和蛇的身体坐标冲突
      55. pSnakeNode cur = ps->_pSnake;
      56. while(cur)
      57. {
      58. if (x == cur->x && y == cur->y)
      59. {
      60. goto again;
      61. }
      62. cur = cur->next;
      63. }
      64. //创建食物的节点
      65. pSnakeNode pFood = (pSnakeNode)malloc(sizeof(SnakeNode));
      66. if (pFood == NULL)
      67. {
      68. return;
      69. }
      70. pFood->x = x;
      71. pFood->y = y;
      72. pFood->next = NULL;
      73. SetPos(x, y);//定义位置
      74. wprintf(L"%lc", FOOD);
      75. ps->_pFood = pFood;
      76. }

      有点小多,请耐心看完哦。

    • 蛇的移步和游戏的初始化,则代码为:
      1. void GameStart(pSnake ps)
      2. {
      3. //1.先设置窗口得大小,再光标隐藏
      4. system("mode con cols = 100 lines = 30");
      5. system("title 贪吃蛇");
      6. HANDLE houtput = GetStdHandle(STD_OUTPUT_HANDLE);
      7. //隐藏光标操作
      8. CONSOLE_CURSOR_INFO CursorInfo;
      9. GetConsoleCursorInfo(houtput, &CursorInfo);//获取控制台光标信息
      10. CursorInfo.bVisible = false; //隐藏控制台光标
      11. SetConsoleCursorInfo(houtput, &CursorInfo); //设置控制台光标状态
      12. //2.打印环境界面和功能介绍
      13. WelcomeToGame();
      14. //3.绘制地图
      15. CreateMap();
      16. //4.创建蛇
      17. InitSnake(ps);
      18. //5.创建食物
      19. CreateFood(ps);
      20. }
      21. void SnakeMove(pSnake ps)
      22. {
      23. //创建一个节点,表示蛇即将到的下一个节点
      24. pSnakeNode pNextNode = (pSnakeNode)malloc(sizeof(SnakeNode));
      25. if (pNextNode == NULL)
      26. {
      27. perror("SnakeMove()::malloc()");
      28. return;
      29. }
      30. switch (ps->_dir)
      31. {
      32. case UP:
      33. pNextNode->x = ps->_pSnake->x;
      34. pNextNode->y = ps->_pSnake->y - 1;
      35. break;
      36. case DOWN:
      37. pNextNode->x = ps->_pSnake->x;
      38. pNextNode->y = ps->_pSnake->y + 1;
      39. break;
      40. case LEFT:
      41. pNextNode->x = ps->_pSnake->x-2;
      42. pNextNode->y = ps->_pSnake->y;
      43. break;
      44. case RIGHT:
      45. pNextNode->x = ps->_pSnake->x+2;
      46. pNextNode->y = ps->_pSnake->y;
      47. break;
      48. }
      49. //检测下一个坐标处是否是食物
      50. if (NextIsFood(pNextNode, ps))
      51. {
      52. EatFood(pNextNode, ps);
      53. }
      54. else
      55. {
      56. NoFood(pNextNode, ps);
      57. }
      58. }

    • 判断下一个坐标是否是食物和下一个位置是食物和下一个位置不是食物,就吃掉食物:则代码为:

      1. int NextIsFood(pSnakeNode pn, pSnake ps)
      2. {
      3. return (ps->_pFood->x == pn->x && ps->_pFood->y == pn->y);
      4. }
      5. void EatFood(pSnakeNode pn, pSnake ps)
      6. {
      7. //头插法
      8. ps->_pFood->next = ps->_pSnake;
      9. ps->_pSnake = ps->_pFood;
      10. //释放下一个位置的节点
      11. free(pn);
      12. pn = NULL;
      13. pSnakeNode cur = ps->_pSnake;
      14. //打印蛇
      15. while (cur)
      16. {
      17. SetPos(cur->x, cur->y);
      18. wprintf(L"%lc", BODY);
      19. cur = cur->next;
      20. }
      21. ps->_score += ps->_food_weight;
      22. //重新创建食物
      23. CreateFood(ps);
      24. }
      25. void NoFood(pSnakeNode pn, pSnake ps)
      26. {
      27. //头插法
      28. pn->next = ps->_pSnake;
      29. ps->_pSnake = pn;
      30. pSnakeNode cur = ps->_pSnake;
      31. while (cur->next->next != NULL)
      32. {
      33. SetPos(cur->x, cur->y);
      34. wprintf(L"%lc", BODY);
      35. cur = cur->next;
      36. }
      37. //把最后一个节点打印成空格
      38. SetPos(cur->next->x, cur->next->y);
      39. printf(" ");
      40. //释放最后一个节点
      41. free(cur->next);
      42. //把倒数第二个节点的地址置为NULL
      43. cur->next = NULL;
      44. }

      确实多,请耐心看完。

    • 游戏运行的逻辑,则代码为:
      1. void GameRun(pSnake ps)
      2. {
      3. //打印帮助信息
      4. PrintHelpInfo();
      5. do
      6. {
      7. //打印总分数和食物的分值
      8. SetPos(64, 10);
      9. printf("总分数:%d\n", ps->_score);
      10. SetPos(64, 11);
      11. printf("当前食物的分数:%d\n", ps->_food_weight);
      12. if (KEY_PRESS(VK_UP) && ps->_dir != DOWN)
      13. {
      14. ps->_dir = UP;
      15. }
      16. else if (KEY_PRESS(VK_DOWN) && ps->_dir != UP)
      17. {
      18. ps->_dir = DOWN;
      19. }
      20. else if (KEY_PRESS(VK_LEFT) && ps->_dir != RIGHT)
      21. {
      22. ps->_dir = LEFT;
      23. }
      24. else if (KEY_PRESS(VK_RIGHT) && ps->_dir != LEFT)
      25. {
      26. ps->_dir = RIGHT;
      27. }
      28. else if (KEY_PRESS(VK_SPACE))
      29. {
      30. pause();
      31. }
      32. else if (KEY_PRESS(VK_ESCAPE))
      33. {
      34. //正常退出游戏
      35. ps->_status = END_NORMAL;
      36. }
      37. else if (KEY_PRESS(VK_F3))
      38. {
      39. //加速
      40. if (ps->_sleep_time > 80)
      41. {
      42. ps->_sleep_time -= 30;
      43. ps->_food_weight += 2;
      44. }
      45. }
      46. else if (KEY_PRESS(VK_F4))
      47. {
      48. //减速
      49. if (ps->_food_weight > 2)
      50. {
      51. ps->_sleep_time += 30;
      52. ps->_food_weight -= 2;
      53. }
      54. }
      55. SnakeMove(ps); //蛇走一步的过程
      56. Sleep(ps->_sleep_time);
      57. } while (ps->_status == OK);
      58. }

    • 暂停响应,检测蛇是否撞墙,检测是否撞到自己和结束游戏 - 善后工作,则代码为:
      1. void pause()
      2. {
      3. while (1)
      4. {
      5. Sleep(200);
      6. if(KEY_PRESS(VK_SPACE))
      7. {
      8. break;
      9. }
      10. }
      11. }
      12. void KillByWall(pSnake ps)
      13. {
      14. if (ps->_pSnake->x == 0 || ps->_pSnake->x == 56 || ps->_pSnake->y == 0 || ps->_pSnake->y == 26)
      15. {
      16. ps->_status = KILL_BY_WALL;
      17. }
      18. }
      19. void KillBySelf(pSnake ps)
      20. {
      21. pSnakeNode cur = ps->_pSnake->next;
      22. while (cur)
      23. {
      24. if (cur->x == ps->_pSnake->x && cur->y == ps->_pSnake->y)
      25. {
      26. ps->_status = KILL_BY_SELF;
      27. break;
      28. }
      29. cur = cur->next;
      30. }
      31. }
      32. void GameEnd(pSnake ps)
      33. {
      34. SetPos(24, 12);
      35. switch (ps->_status)
      36. {
      37. case END_NORMAL:
      38. wprintf(L"你主动退出\n");
      39. break;
      40. case KILL_BY_WALL:
      41. wprintf(L"您撞上自己了 ,游戏结束!\n");
      42. break;
      43. case KILL_BY_SELF:
      44. wprintf(L"您撞墙了,游戏结束!\n");
      45. break;
      46. }
      47. //释放蛇身的链表
      48. pSnakeNode cur = ps->_pSnake;
      49. while (cur)
      50. {
      51. pSnakeNode del = cur;
      52. cur = cur->next;
      53. free(del);
      54. }
      55. }

    • 贪吃蛇的主函数代码确实多,但也是最重要一部分。请大家耐心看完哦。

  • 相关阅读:
    iptables服务简单使用
    “蔚来杯“2022牛客暑期多校训练营7 A题: Floor Tiles in a Park
    什么是应变能力?如何提高应变能力?
    第3章业务功能开发(实现显示线索主页面,并查询表单各个下拉框数据)
    多路混流实操流程
    28 mysql 数据记录的 存储更新删除
    【LeetCode】94. 二叉树的中序遍历 [ 左子树 根结点 右子树 ]
    思科设备BGP配置命令
    nginx下载安装和日志切割
    Linux后台运行程序
  • 原文地址:https://blog.csdn.net/J15212630255/article/details/138137141