C语言写个贪吃蛇游戏

贪吃蛇是个非常经典的游戏,用C语言来实现也是一个好玩的事情。这个游戏我写完后放在知乎,竟然点赞的人数超级多。我觉得大家喜欢,一个方面是因为写得简单,大家都能看得懂,一个可扩展性还是非常强的。

我试了说一下这个代码 核心的三个函数

    menu();setup();draw();

menu用来设置菜单,也就是我们一运行看到的那个。setup用来设置参数,我们需要设置高度和宽度,还有分数,食物的位置。draw也就是画,也就是画整个画面。

还有一个枚举类型 这个结构体用来设置蛇的几个状态,我觉得这个也是面向对象编程的一个思想,把蛇的状态都封装成一个枚举类型。

typedef enum
{STOP = 0,LEFT,RIGHT,UP,DOWN
}Direction;

还有

/*判断贪吃蛇的长度*/
void logic()

这个函数,这个函数应该是整个贪吃蛇的精髓了,要理解代码怎么把蛇给连接起来。用了点巧妙的东西。

来看这里面的关键代码

/*把上一个位置记下*/int lastX = tailX[0];int lastY = tailY[0];int last2X, last2Y;/*重新获取当前的位置*/tailX[0]=x;tailY[0]=y;int i=0;/*遍历整条蛇的长度 把 0 的位置空出来,其余蛇的位置往后面的空间移动*/for(i=1; i<ntail;i++){last2X = tailX[i];last2Y = tailY[i];tailX[i]=lastX;tailY[i]=lastY;lastX = last2X;lastY = last2Y;}

lastX lastY 用来存上一次的蛇头的位置。后面的 for 循环,通过tail 蛇的长度,把蛇上个状态给保存到数组tailX tailY里面。

完整代码


#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <Windows.h>
#include <time.h>
#include <conio.h> /*键盘输入获取*/bool gameOver;
bool stop = false;
bool hit = false;
/*游戏的边框大小*/
const int width = 50;
const int height = 20;
/*蛇的坐标,食物的坐标还有分数*/
int x,y,fruitX,fruitY,score;
/*蛇每个点的坐标*/
int tailX[200],tailY[200];
/*蛇的默认长度*/
int ntail=3;typedef enum
{STOP = 0,LEFT,RIGHT,UP,DOWN
}Direction;Direction Dir;
/*开始菜单*/
void menu()
{int a;printf("------------------------------------------------------------------\n");printf("|                              贪吃蛇游戏                        |\n");printf("|                              1) 新游戏                         |\n");printf("|                              2) 开始边界                       |\n");printf("|                              3) 退出游戏                       |\n");printf("------------------------------------------------------------------\n");printf("---->请输入你的选择:");scanf("%d", &a);
}/*初始化状态*/
void setup()
{gameOver = false;/*根据当前时间设置“随机数种子”*/srand(time(NULL));Dir = STOP;/*贪吃蛇的位置,固定在中间*/x= width/2;y= height/2;/*食物的位置,位置是随机的*/fruitX = rand()%width;fruitY = rand()%height;score = 0;
}
/*绘制界面*/
void draw()
{if(stop == true){return;}system("cls");/*清除屏幕*/printf("分数:%d",score);printf("\n");/*第一行*/int i;for(i= 0 ;i<width+1;i++){printf("-");}printf("\n");/*画中间的画面*/int p;for(p= 0 ;p<height;p++)/*高度*/{int q;for(q= 0 ;q<width;q++)/*宽度*/{/*第一行最后已给字符*/if(q==0 || q==width-1){printf("|");}if(p == fruitY && q == fruitX)/*食物的随机坐标*/{printf("O");}else{int k=0;bool print = false;/*贪吃蛇的长度 默认长度是 3*/for(k=0;k<ntail;k++){if(tailX[k]==q && tailY[k]==p){printf("*");print = true;}}/*如果这个位置打印了 * 就不要打印空格了*/if(!print){printf(" ");}}}printf("\n");}/*最后一行*/int j;for(j= 0 ;j<width+1;j++){printf("-");}}
/*按键输入控制*/
void input()
{if(_kbhit()){/*获取键盘的输入字符*/switch(_getch()){case '4':case 75:/*左键*/Dir = LEFT;hit= true;break;case '8':case 72:/*上键*/Dir = UP;hit= true;break;case '6':case 77:/*右键*/Dir = RIGHT;hit= true;break;case '2':case 80:/*向下键盘键 */Dir = DOWN;hit= true;break;case 'x':case 27:/*ESE*/gameOver = true;break;case 32:/*空格 暂停键*/stop = !stop;break;}}else if(!hit && stop == false)/*如果没有改变方向*/{x++;}
}
/*判断贪吃蛇的长度*/
void logic()
{if(stop == true){return;}/*把上一个位置记下*/int lastX = tailX[0];int lastY = tailY[0];int last2X, last2Y;/*重新获取当前的位置*/tailX[0]=x;tailY[0]=y;int i=0;/*遍历整条蛇的长度 把 0 的位置空出来,其余蛇的位置往后面的空间移动*/for(i=1; i<ntail;i++){last2X = tailX[i];last2Y = tailY[i];tailX[i]=lastX;tailY[i]=lastY;lastX = last2X;lastY = last2Y;}/*根据方向来改变x y 的值*/switch(Dir){case UP:y--;break;case DOWN:y++;break;case LEFT:x--;break;case RIGHT:x++;break;}if(x<0 || width<x || y<0 || height<y){gameOver = true;/*清除屏幕*/system("cls");printf("------------------------------------------------------------------\n");printf("|                                                                |\n");printf("|                                                                |\n");printf("|                             游戏结束                           |\n");printf("|                                                                |\n");printf("|                                                                |\n");printf("------------------------------------------------------------------\n");}if(x==fruitX && y==fruitY){/*吃了一个食物,蛇的长度增加1*/ntail++;score+=10;/*更新下一个食物的位置*/fruitX = rand()%width;fruitY = rand()%height;}
}
int main()
{
#if 0while(1){printf("%d\n",_getch());}
#endifmenu();setup();draw();/*循环画贪吃蛇的界面*/while(!gameOver){draw();input();logic();Sleep(70);}return 0;
}

上面这段代码直接在Dev C++上面应该是可以运行的,很多人在知乎上私信问我,为什么我的贪吃蛇执行不了呢,可能就是平台不同,少了这个头文件,少了那个头文件,但是你为什么不能跟我一样,用Dev C++呢,轻量级,简单。代码的精髓是什么?我认为精髓一定是思想,不是你写了多少行代码,用了什么高端的IDE。

我自认为我的注释已经写得不错了,所以就没有什么好说明的了吧,有不明白的把代码过一下,至于屏幕刷新这个东西,如果只是用时间刷新就会闪屏,所以出现了一个双缓存,把要显示的东西送到一个buff里面去,另一个buff用来显示,这样就可以保证不会出现闪屏。除了写贪吃蛇,可以用这个方法写其他小程序,挺有意思的。

在知乎上,发起了一个C语言 100 行代码之内实现贪吃蛇的问题。我觉得很不错,里面很多同学的回复都非常赞,特别是叶大神的回复。

https://www.zhihu.com/question/360814879/answer/1013986215


扫码或长按关注

回复「 篮球的大肚子」进入技术群聊

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/467954.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

seir模型matlab_疫情专题 | 传染病的经典数学模型

在此次新冠肺炎疫情防控过程中&#xff0c;对疫情发展趋势的科学预测显得尤为重要。而这背后&#xff0c;离不开对传染病传播规律的建模。今天&#xff0c;小编就带各位数学学子们来了解一下传染病的四大经典数学模型&#xff1a;SI/SIS/SIR/SEIR。其中用到了许多微分方程的知识…

vlh 标签详解

1.vlh:root root标签做为所有vlh标签的根标签. 1)value 在给定的范围内&#xff0c;包含在ValueList或list的变量名. List的实例自动被DefaultListBackedValueList包装在ValueList中 2)id 如果有多个表被包含在一个request中&#xff0c;ID属性能区分每个表。id被追加到所…

ubuntu电脑安装硬盘

最近在做安卓开发&#xff0c;一套RK3399的安卓代码&#xff0c;解压编译后占用170多G的硬盘。所以呢&#xff0c;原来1T大小的硬盘&#xff0c;很快就沾满了&#xff0c;然后我赶紧给北京总部申请买了一个新的硬盘。现在的台式电脑都是用的SATA硬盘接口&#xff0c;我赶紧就上…

python 制作抽奖箱_用Excel函数制作抽奖箱

话说各在公司每年的年会上&#xff0c;或者平时的一些分组活动上&#xff0c;又或者是某个内部组织的业务竞赛上……偶尔会遇到抽奖或抽签的环节&#xff0c;例如你的公司开展了一个实操的业务竞赛&#xff0c;一共80道题目&#xff0c;参赛人员要随机抽取题目然后做答。好的&a…

字典表

字典表 &#xff1a; dict 声明 键;值dict(键值) 操作 获取d.get(键‘默认值) 合并d.update(d2) 键值emp.items 菜单emp.keys 效果emp.values得到视图 遍历打印for x in emp.keys 支持嵌套 由于哈希算法导致顺序混乱可以将方法转换为列表&#xff0c;在排序 方法二 全局函…

Sql Server常用函数及技巧

使用Sql Server好长时间了&#xff0c;今天特别想总结一下&#xff0c;算是回顾吧&#xff01; 总结&#xff1a; 其实很多技巧&#xff0c;都是基于SQL Server自带的System Views&#xff0c;System Stored Procedures&#xff0c;System Functions (常用函数都在在里面)。 常…

十分钟让你明白AIDL

前言我在[003]AIDL是什么中介绍的AIDL&#xff0c;但是好像还有朋友不明白问我&#xff0c;那我就来写一个终极版的文章&#xff0c;让你十分钟彻底明白AIDL&#xff0c;以下代码全为手写。目标Server进程注册一个Binder服务到SM&#xff0c;该Binder服务提供两个接口&#xff…

表达式

表达式与分支 语句 分割文档main spilit 风格pep8 赋值 序列赋值 列表切割法 扩展序列解包赋值 *获取剩余 多目标赋值 数字256为界以内指向同对象 以外就不同 字符串3个 列表属于引用类型 不要共同引用两种方法 参数化赋值 列表也可以进行参数化赋值 表达式 函数 方法…

Linux下故障分析方法

1、背景有时候会遇到一些疑难杂症&#xff0c;并且监控插件并不能一眼立马发现问题的根源。这时候就需要登录服务器进一步深入分析问题的根源。那么分析问题需要有一定的技术经验积累&#xff0c;并且有些问题涉及到的领域非常广&#xff0c;才能定位到问题。所以&#xff0c;分…

TEXT宏

TEXT宏是windows程序设计中经常遇到的宏&#xff0c;定义在 <winnt.h>中 TCHAR *P TEXT("this is a const string"); 如果使用UNICODE字符集&#xff0c;则TEXT&#xff08;“....”&#xff09; &#xff0c;相当于 L"....." &#xff1b; 如果使…

作业题

import random fopen(‘data.txt’,‘w’) for i in range(10000): f.write(str(random.randint(1,100))) f.write(’\n’) f.seek(0) print(f.read()) f.close f.seek()函数 移动文件中n个操作 正为向结束方向 enumerate() 函数用于将一个可遍历的数据对象(如列表、元组或字符…

即将放弃python的app_python放弃之 模块和包

importprint(frrom the my_module.py)money1000def rend1():print(my_my_module->reand1->money,money)def rend2():print(my_module->read2 calling read1)read1()def change():global moneymoney0模块可以包含可执行语句和函数的定义&#xff0c;这些语句的目的是初…

ubuntu 安装gitlab

gitlab 类似github&#xff0c;可以用来管理代码。当然除了他们两个还有很多代码管理的工具&#xff0c;国内的也有。我这篇文章就只讲gitlab的安装过程。但是gitlab并不是轻量级的东西&#xff0c;占用大概4~8G的内存&#xff0c;特别是merge的代码比较多的时候&#xff0c;占…

一些常用的linux命令(2)

参考&#xff1a;http://www.cnblogs.com/laov/p/3541414.html 系统管理命令 stat 显示指定文件的详细信息&#xff0c;比ls更详细 who 显示在线登陆用户 whoami 显示当前操作用户 hostname 显示主机名 uname 显示系统信…

迭代

迭代 文章目录迭代内置可迭代对象 range&#xff08;&#xff09;内置函数使用 map&#xff08;函数加对象&#xff09;函数定义与参数定义传参作用域函数参数传值可迭代对象支持迭代协议 遍历循环 因为 迭代协议 方法—next—&#xff08;&#xff09;函数next消耗内存小所占空…

python os renames_Python3 os.renames() 方法

概述os.renames() 方法用于递归重命名目录或文件。类似rename()。语法renames()方法语法格式如下&#xff1a;os.renames(old, new)参数old -- 要重命名的目录new --文件或目录的新名字。甚至可以是包含在目录中的文件&#xff0c;或者完整的目录树。返回值该方法没有返回值实例…

普通人的节奏

在立春当夜&#xff0c;这南方城市的上空响起了2020年的第一声春雷&#xff0c;大家都期待着这一声霹雳除去一切霉噩&#xff0c;带来万象更新。之后的天气逐渐变好&#xff0c;近日来更是风和日丽。前日&#xff0c;去采购食品和日用品&#xff0c;久不出门&#xff0c;趁好天…

DDD领域模型自动生成?

我不想写代码&#xff0c;只想通过界面配置一下就能自动生成代码多好&#xff0c;如果界面也能自动生成&#xff0c;那就更好了转载于:https://www.cnblogs.com/PerfectBeauty/p/7351849.html

函数与lambda

文章目录参数匹配将形参赋值&#xff08;关键字匹配&#xff09;默认位置匹配解包在元组前加*函数传递字典表&#xff0c;使用**解包字典表两种声明法lambda表达式初学者逻辑采用字典表可以实现委托map函数把列表对象每一个拿出来放入函数中进行迭代后放入新的列三种方式filter…

还没学python_2个月过去了!还没学会python?用《流畅的python》15天带你学会

学了很久python&#xff0c;还是完全没有思路自己学了很久的python&#xff0c;还是感觉没学到什么&#xff0c;出现这样的情况的python学习者有很多&#xff0c;只是大多都是放弃了&#xff01;出现这样的原因主要有以下三点&#xff1a;1、没有人带领&#xff0c;遇到问题解决…