c语言实战之贪吃蛇
文章目录
- 前言
- 效果展示
- 游戏用到的图片
- 游戏思路一览
- 游戏前准备
- 一、贪吃蛇、食物、障碍物节点坐标的结构体
- 二、枚举游戏状态、和贪吃蛇的方向
- 三、维护运行的结构体
- 游戏开始前的初始化
- 一、学习图形库相关知识
- 二、设置背景
- 三、欢迎界面
- 四、初始化贪吃蛇
- 五、生成障碍物
- 六、生成食物
- 七、游戏前的初始化
- 游戏运行过程
- 一、打印分数和显示当前食物分数
- 二、检测按键输入
- 三、的移动过程
- 四、游戏运行
- 游戏结束
前言
1、使用工具:vs2022、EasyX图形库。
2、面向对象:非常适用于刚学完c语言和学过单链表的小伙伴哦。
3、作用:能够提高学习编程的兴趣、复习学过的c语言和单链表、提高编程的能力和逻辑能力。
效果展示
贪吃蛇
游戏用到的图片
背景:
蛇节点:
食物:
障碍物:
图片来自网络。
当然也可以改成其他图片哦。
游戏思路一览
游戏前准备
一、贪吃蛇、食物、障碍物节点坐标的结构体
1、因为要包含不同类型所以我们用结构体。
2、我们用单链表来将贪吃蛇连在一起,不同位置的食物、障碍物也是通过单链表连在一起,这样方便我们找到它们的位置。
3、结构体成员:记录它们的坐标:(x,y),和记录下一个位置的前驱结构体指针:next。
4、参考代码:
//蛇的节点和食物
typedef struct SnakesNode
{//坐标int x;int y;//链接struct SnakesNode* next;
}SnakesNode, *pSnakesNode;
二、枚举游戏状态、和贪吃蛇的方向
1、游戏状态:正常运行、正常结束、撞墙、咬到自己、撞到障碍物。
2、游戏状态的作用:通过游戏状态判断蛇下一步应该怎么走。
3、参考代码:
//游戏状态
enum GAME_STATUS
{OK,//正常运⾏KILL_BY_WALL,//撞墙KILL_BY_SELF,//咬到⾃⼰KILL_BY_BARR,//撞到障碍物了END_NOMAL//正常结束
};
4、贪吃蛇的方向:上下左右。
5、方向的作用:判断贪吃蛇下一步的方向。
6、参考代码:
//⽅向
enum DIRECTION
{UP = 1,//上DOWN,//下LEFT,//左RIGHT//右
};
三、维护运行的结构体
结构体成员:
维护整条贪吃蛇的指针、维护⻝物的指针、维护障碍物的指针、贪吃蛇头的方向默认是向右、游戏状态、当前获得分数、当前食物的分数、每走⼀步休眠时间。
//维护整个运行
typedef struct Snakes
{pSnakesNode _pSnake;//维护整条贪吃蛇的指针pSnakesNode _pFood;//维护⻝物的指针pSnakesNode _pBarrier;//维护障碍物的指针enum DIRECTION _Dir;//蛇头的⽅向默认是向右enum GAME_STATUS _Status;//游戏状态int _Socre;//当前获得分数int _foodWeight;//默认每个⻝物10分int _SleepTime;//每⾛⼀步休眠时间
}Snakes,*pSnakes;
游戏开始前的初始化
一、学习图形库相关知识
在初始化之前我们先来了解一下图形库中的函数,如果没有的话可以先下载EasyX,这个直接在浏览器上搜索下载就行了,这个图形库会自己适配vs2022,并且里面有很多关于图形库的函数,感兴趣的可以去了解一下哦
界面:
关于坐标:
我们现在来了解我们制作贪吃蛇用到的一些函数
使用下面函数都要包含头文件:#include<graphics.h>
1、loadimage函数
作用:获取图片到 IMAGE(图像对象的类型) 指针里。
// 从图片文件获取图像(bmp/gif/jpg/png/tif/emf/wmf/ico)
void loadimage(IMAGE* pDstImg, // 保存图像的 IMAGE 对象指针LPCTSTR pImgFile, // 图片文件名int nWidth = 0, // 图片的拉伸宽度int nHeight = 0, // 图片的拉伸高度bool bResize = false // 是否调整 IMAGE 的大小以适应图片
);
参数说明:
这个函数报错问题:
更改以下设置就可以了:
2、putimage函数
作用:将图片加载到设备上。
// 绘制图像
void putimage(int dstX, // 绘制位置的 x 坐标int dstY, // 绘制位置的 y 坐标IMAGE *pSrcImg, // 要绘制的 IMAGE 对象指针DWORD dwRop = SRCCOPY // 三元光栅操作码
);
3、outtextxy函数
作用:在指定坐标输出字符串。
void outtextxy(//坐标int x,int y,LPCTSTR str //字符串
);
4、initgraph函数
作用:初始化窗口大小。
HWND initgraph(int width,int height,int flag = NULL
);
二、设置背景
1、我们背景窗口设置为1000,640,游戏的范围为800,640。
2、多出来的范围用于显示分数。
//绘制窗口大小
initgraph(1000, 640);
//图像对象
IMAGE bg_img;
//设置背景
loadimage(&bg_img, "blackground.bmp", 800, 640, true);
//加载到设备
putimage(0, 0, &bg_img);
效果:
三、欢迎界面
1、我们先在中间打印,然后再重新加载一个背景将前面的覆盖,再打印按键的提示,最后再覆盖一次,为后面的游戏准备。
//欢迎界⾯
void WelcomeToGame(IMAGE *bg_img)
{//在中间打印outtextxy(340, 300, "欢迎进入游戏...");//在末尾提示outtextxy(0, 620, "请按任意键开始游戏...");//任意键继续的一个函数system("pause");//重新加载背景putimage(0, 0, bg_img);//游戏按键提示outtextxy(190, 300, "⽤ ↑ . ↓ . ← . → 分别控制蛇的移动, F3为加速,F4为减速");outtextxy(0, 620, "请按任意键开始游戏...");system("pause");//再重新加载背景putimage(0, 0, bg_img);
}
效果:
四、初始化贪吃蛇
1、通过链表来将贪吃蛇连起来,默认为5个节点。
2、贪吃蛇的初始位置:(0,0),贪吃蛇一个节点的大小,我们设置为20*20
用于个宏定义来设置,方便以后更改:
//贪吃蛇的初始位置
#define POS_X 0
#define POS_Y 0
//节点大小
#define SIZE 20
3、链接过程:先创建一个头节点然后再用头插法将其他的节点头插进去,最后将头节点赋给维护贪吃蛇的指针,其他节点的位置就是与前一个节点的x隔着一个SIZE的大小(因为默认向右)。
4、再创建过程顺便将图片加载出来。
5、代码参考:
//加载贪吃蛇节点的函数
void Node(pSnakesNode p)
{//函数对象IMAGE node;//获取图片信息loadimage(&node, "屏幕截图 2024-01-28 140514.png", SIZE, SIZE, true);//加载到设备上putimage(p->x, p->y, &node);
}
//初始化贪吃蛇身
void InitSnake(pSnakes ps)
{//头节点pSnakesNode tmp = (pSnakesNode)malloc(sizeof(SnakesNode));//判断是否成功assert(tmp);//初始化第一个节点tmp->next = NULL;tmp->x = POS_X;tmp->y = POS_Y;//将这个节点加载到设备上Node(tmp);//有头插法将其他4个节点加入for (int i = 1; i <=4; i++){pSnakesNode tmp1 = (pSnakesNode)malloc(sizeof(SnakesNode));assert(tmp1); tmp1->x = POS_X+i*SIZE;tmp1->y = POS_Y;//新的接后的一个tmp1->next = tmp;tmp = tmp1;tmp1 = NULL;Node(tmp);}//将结构体中维护蛇节点的指针赋值ps->_pSnake = tmp;tmp = NULL;//其他属性初始化ps->_Dir = RIGHT;ps->_Socre = 0;ps->_Status = OK;ps->_SleepTime = 150;ps->_foodWeight = 10;ps->_pBarrier = NULL;
}
效果:
五、生成障碍物
1、开始设置10个障碍物。
2、利用链表将所有障碍物连起来(通过头插入),这样方便找到它们的位置。
3、生成的位置:不能在贪吃蛇的蛇身、要在游戏的地图范围内,并且要是SIZE的倍数(保证与贪吃蛇的位置在同一个水平上)。
4、参考代码:
//加载障碍物图片
void Barr(pSnakesNode p)
{IMAGE barrier;loadimage(&barrier, "屏幕截图 2024-01-28 165936.png", SIZE, SIZE, true);putimage(p->x ,p->y, &barrier);
}
//生成障碍物
void barrier(pSnakes ps)
{//障碍物的坐标int x = 0;int y = 0;pSnakesNode p;//生成障碍物的坐标for (int i = 0; i < 10; i++){ again://跳回do {//1、要在地图内x = rand() % 780 + 1;y = rand() % 620 + 1;//2、x和y要在SIZE的倍数上} while (x % SIZE != 0 || y % SIZE != 0);//3、不能出现在贪吃蛇的身上pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){if (tmp->x == x && tmp->y == y){//如果碰到是在贪吃蛇的身上的话就跳转到开始的位置goto again;}tmp = tmp->next;}p = (pSnakesNode)malloc(sizeof(SnakesNode));assert(p);p->next = NULL;p->x = x;p->y = y;//加载障碍物图片Barr(p);//如果ps->_pBarrier == NULL则将p赋给ps->_pBarrier,其他的用头插插入if (ps->_pBarrier == NULL){ps->_pBarrier = p;}else{p->next = ps->_pBarrier;ps->_pBarrier = p;p = NULL;} }
}
六、生成食物
1、开始设置10个食物。
2、利用链表将所有食物连起来(通过头插入),这样方便找到它们的位置。
3、生成的位置:不能在贪吃蛇的蛇身、要在游戏的地图范围内,并且要是SIZE的倍数(保证与贪吃蛇的位置在同一个水平上),不要在障碍物上。
4、参考代码:
//加载食物图片到设备上
void Foob(pSnakesNode p)
{IMAGE food;loadimage(&food, "屏幕截图 2024-01-28 151002.png", SIZE, SIZE, true);putimage(p->x, p->y, &food);
}
//创建单个食物
pSnakesNode Create(pSnakes ps)
{
//食物坐标pSnakesNode p;int x = 0;int y = 0;
again:
in:do {//1、要在地图内x = rand() % 780 + 1;y = rand() % 620 + 1;//2、是SIZE的倍数} while (x % SIZE != 0 || y % SIZE != 0);//3、不能出现在贪吃蛇的身上pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){if (tmp->x == x && tmp->y == y){goto again;}tmp = tmp->next;}//4、不能出现在障碍物身上pSnakesNode tmp1 = ps->_pBarrier;while (tmp1 != NULL){if (tmp1->x == x && tmp1->y == y){goto in;}tmp1 = tmp1->next;}p = (pSnakesNode)malloc(sizeof(SnakesNode));assert(p);p->next = NULL;p->x = x;p->y = y;//加载图片Foob(p);//返回这个食物节点return p;
}
//生成10个食物
void CreateFood(pSnakes ps)
{pSnakesNode p;//通过循环进行头插for (int i = 0; i < 10; i++){pSnakesNode p = Create(ps);if (ps->_pFood == NULL){ps->_pFood= p;}else{p->next = ps->_pFood;ps->_pFood = p;p = NULL;}}
}
5、生成障碍物和食物效果
七、游戏前的初始化
void GameStart(pSnakes ps)
{//绘制窗口大小initgraph(1000, 640);//图像对象IMAGE bg_img;//设置背景loadimage(&bg_img, "blackground.bmp", 800, 640, true);//加载到设备putimage(0, 0, &bg_img);//欢迎界面WelcomeToGame(&bg_img);// 初始化InitSnake(ps);//生成障碍物barrier(ps);//创建⻝物CreateFood(ps);
}
游戏运行过程
一、打印分数和显示当前食物分数
我们之前提到过在设置窗口的时候设置了比游戏地图大,因此我们就在多的位置上显示获得的分数和当前食物的得分。
//记录的分
char str[10] = { 0 };
//利用该函数将当前得分输入到str上
sprintf(str, "%d", ps->_Socre);
//记录当前食物的分数
char str1[5] = { 0 };
sprintf(str1, "%d", ps->_foodWeight);
//在规定的位置上输出
outtextxy(810, 20, "当前得分:");
outtextxy(940, 20, str);
outtextxy(810, 80, "当前食物得分:");
outtextxy(940, 80, str1);
效果:
二、检测按键输入
1、利用GetAsyncKeyState函数检测按键
头文件:#include<Windows.h>
函数原型:
SHORT GetAsyncKeyState([in] int vKey
);
将键盘上每个键的虚拟键值传递给函数,函数通过返回值来分辨按键的状态。
虚拟键值参考
GetAsyncKeyState 的返回值是short类型,在上⼀次调用 GetAsyncKeyState 函数后,如果返回的16位的short数据中,最高位是1,说明按键的状态是按下,如果最⾼是0,说明按键的状态是抬起;如果最低位被置为1则说明,该按键被按过,否则为0。
如果我们要判断⼀个键是否被按过,可以检测GetAsyncKeyState返回值的最低值是否为1。
因此我们可以定义一个宏来判断按键是否被按过
//按键
#define KEY_PRESS(VK) ((GetAsyncKeyState(VK)&0x1) ? 1 : 0)
这样我们就可以通过按键控制贪吃蛇的状态了。
2、按键:
⽤ ↑ . ↓ . ← . → 分别控制蛇的移动, F3为加速,F4为减速,空格为暂停,ESC为正常退出
暂停:
利用死循环来控制暂停(通过Sleep函数来延迟),当再次接受到空格键时直接打破。
//暂停
void pause()
{while (1){ //延迟函数Sleep(300);if (KEY_PRESS(VK_SPACE)){break;}}
}
加速减速:
通过减少延迟的时间就可以加速,减少就反之,当然加速每个食物的分数就会增加,减速每个食物的分数就会减少。
通过选择语句来判断
//判断按键
//上,走上就不能走下了,防止重叠
if (KEY_PRESS(VK_UP) && ps->_Dir != DOWN)
{//改变状态ps->_Dir = UP;
}
//下
else if (KEY_PRESS(VK_DOWN) && ps->_Dir != UP)
{//改变状态ps->_Dir = DOWN;
}
//左
else if (KEY_PRESS(VK_LEFT) && ps->_Dir != RIGHT)
{//改变状态ps->_Dir = LEFT;
}
//右
else if (KEY_PRESS(VK_RIGHT) && ps->_Dir != LEFT)
{//改变状态ps->_Dir = RIGHT;
}
//空格
else if (KEY_PRESS(VK_SPACE))
{
//暂停pause();
}
//ESC
else if (KEY_PRESS(VK_ESCAPE))
{ //改变游戏状态ps->_Status = END_NOMAL;break;
}
//F3
else if (KEY_PRESS(VK_F3))
{//控制延迟的时间if (ps->_SleepTime >= 50){ps->_SleepTime -= 20;//分数增加ps->_foodWeight += 2;}
}
//F4
else if (KEY_PRESS(VK_F4))
{//控制延迟的时间if (ps->_SleepTime < 350){ps->_SleepTime += 20;//分数减少ps->_foodWeight -= 2;//不能太慢if (ps->_SleepTime == 350){ps->_foodWeight = 1;}}
}
三、的移动过程
第一步获得下一步的坐标:
//计算下一个坐标的位置
// //下一个节点
pSnakesNode tmp = (pSnakesNode)malloc(sizeof(SnakesNode));
assert(tmp);
tmp->next=NULL;
//上
if (ps->_Dir == UP)
{tmp->y = ps->_pSnake->y - SIZE;tmp->x = ps->_pSnake->x;
}
//下
else if (ps->_Dir == DOWN)
{tmp->y = ps->_pSnake->y + SIZE;tmp->x = ps->_pSnake->x;
}
//左
else if (ps->_Dir == LEFT)
{tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x-SIZE;
}
//右
else if (ps->_Dir == RIGHT)
{tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x+SIZE;
}
第二步判断是不是食物:
通过下一步的坐标和食物的坐标对比,如果符合的话就证明是食物
//判断是不是食物
bool judgment(pSnakesNode psn, pSnakes ps)
{pSnakesNode tmp = ps->_pFood;//遍历食物的所有坐标while (tmp != NULL){if (tmp->x == psn->x && tmp->y == psn->y){return true;}tmp = tmp->next;}return false;
}
第三步下一步是食物:
1、将下一步的坐标和贪吃蛇链接,并重新加载这个贪吃蛇节点图片。
2、重新生成一个新食物并重新加载食物节点图片。
3、利用前后指针的方法找到该食物的坐标和该食物的上一个坐标。
3、如果该食物是头节点的话,保存下一个节点同时释放该食物节点,用新食物做食物的头节点并链接保存的节点。
4、如果不是的话就保存该食物节点的下一个节点,用新食物的next连这个节点,再用保存的该食物的上一个节点的next连新食物,最后释放该食物节点。
5、进行加分。
//下⼀个节点是⻝物
void EatFood(pSnakesNode psn, pSnakes ps)
{//直接将下一个坐标和当前的贪吃蛇链接,并加载图片psn->next = ps->_pSnake;ps->_pSnake = psn;Node(psn);//找食物节点//前pSnakesNode p = ps->_pFood;//后pSnakesNode p1 =NULL;while (p != NULL){if (p->x == psn->x && p->y == psn->y){break;}p1 = p;p = p->next;}//生成一个食物pSnakesNode p2 = Create(ps);//头if (p1 == NULL){p2->next = p->next;free(p);ps->_pFood = p2;}else{p1->next = p2;p2->next = p->next;free(p);}//加分ps->_Socre += ps->_foodWeight;
}
第四步下一步不是食物:
1、链接。
2、找到倒数第二个节点,在这个过程顺便加载新的贪吃蛇节点图片。
3、通过倒数第二个节点,将倒数第一个节点位置加载为背景颜色然后释放最后一个节点。
//下一个节点不是食物
void NoFood(pSnakesNode psn, pSnakes ps)
{//链接psn->next = ps->_pSnake;ps->_pSnake = psn;pSnakesNode tmp = ps->_pSnake;//找倒数第二个节点while (tmp->next->next!= NULL){Node(tmp);tmp = tmp->next;}IMAGE b;//设置背景loadimage(&b, "blackground.bmp", SIZE, SIZE, true);putimage(tmp->next->x, tmp->next->y, &b);//释放最后一个free(tmp->next);tmp->next = NULL;
}
第五步判断是否撞墙:
通过判断蛇头的位置的坐标是否超出地图的范围,超过了就改变蛇的状态。
//撞墙检测
int KillByWall(pSnakes ps)
{
//判断蛇头的位置是否超出地图的范围if (ps->_pSnake->x<0 || ps->_pSnake->x>780 || ps->_pSnake->y<0|| ps->_pSnake->y>620){//改变贪吃蛇的状态为撞墙ps->_Status = KILL_BY_WALL;return 1;}else{return 0;}
}
第六步判断是否撞到障碍物
通过遍历障碍物的坐标与蛇头坐标对比。
//撞到障碍物
int Hittheobstacles(pSnakes ps)
{pSnakesNode tmp = ps->_pSnake;pSnakesNode tmp1 = ps->_pBarrier;//遍历while (tmp1 != NULL){if (tmp1->x == tmp->x && tmp1->y == tmp->y){//改变游戏运行状态为撞到障碍物ps->_Status = KILL_BY_BARR;return 1;}tmp1 = tmp1->next;}return 0;
}
第七步判断是否咬到自己了
通过从贪吃蛇的第二个节点开始遍历与蛇头坐标对比。
/撞自⾝检测
int KillBySelf(pSnakes ps)
{pSnakesNode tmp = ps->_pSnake->next;while (tmp){//判断if (tmp->x == ps->_pSnake->x && tmp->y == ps->_pSnake->y){//改变游戏运行状态ps->_Status= KILL_BY_SELF;return 1;}tmp = tmp->next;}return 0;
}
最后总的贪吃蛇移动的代码:
//贪吃蛇的移动
void SnakeMove(pSnakes ps)
{//计算下一个坐标的位置// //下一个节点pSnakesNode tmp = (pSnakesNode)malloc(sizeof(SnakesNode));assert(tmp);tmp->next=NULL;//上if (ps->_Dir == UP){tmp->y = ps->_pSnake->y - SIZE;tmp->x = ps->_pSnake->x;}//下else if (ps->_Dir == DOWN){tmp->y = ps->_pSnake->y + SIZE;tmp->x = ps->_pSnake->x;}//左else if (ps->_Dir == LEFT){tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x-SIZE;}//右else if (ps->_Dir == RIGHT){tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x+SIZE;}//判断是否是食物//是食物if (judgment(tmp,ps)){ EatFood(tmp, ps);}//不是食物else{NoFood(tmp, ps);}//是否撞墙KillByWall(ps);//是否咬到自己KillBySelf(ps);//是否撞到障碍物Hittheobstacles(ps);
}
四、游戏运行
1、通过do while循环,至少有一次的按键机会。
2、while()来判断游戏状态,如果不是OK就代表结束游戏了。
3、rhythm是全局变量,用于控制游戏节奏,每走一次就加1,默认为0,当每走到一定步数时,速度就会加快,同时每个食物的分数增加,还有当走到一定步数时障碍物会重新刷新。
//游戏运⾏
void GameRun(pSnakes ps)
{do {//记录的分char str[10] = { 0 };//利用该函数将当前得分输入到str上sprintf(str, "%d", ps->_Socre);//记录当前食物的分数char str1[5] = { 0 };sprintf(str1, "%d", ps->_foodWeight);//在规定的位置上输出outtextxy(810, 20, "当前得分:");outtextxy(940, 20, str);outtextxy(810, 80, "当前食物得分:");outtextxy(940, 80, str1);if (KEY_PRESS(VK_UP) && ps->_Dir != DOWN){ps->_Dir = UP;}else if (KEY_PRESS(VK_DOWN) && ps->_Dir != UP){ps->_Dir = DOWN;}else if (KEY_PRESS(VK_LEFT) && ps->_Dir != RIGHT){ps->_Dir = LEFT;}else if (KEY_PRESS(VK_RIGHT) && ps->_Dir != LEFT){ps->_Dir = RIGHT;}else if (KEY_PRESS(VK_SPACE)){pause();}else if (KEY_PRESS(VK_ESCAPE)){ps->_Status = END_NOMAL;break;}else if (KEY_PRESS(VK_F3)){if (ps->_SleepTime >= 50){ps->_SleepTime -= 20;ps->_foodWeight += 2;}}else if (KEY_PRESS(VK_F4)){if (ps->_SleepTime < 350){ps->_SleepTime += 20;ps->_foodWeight -= 2;if (ps->_SleepTime == 350){ps->_foodWeight = 1;}}}//暂停Sleep(ps->_SleepTime);//移动蛇SnakeMove(ps);//控制节奏rhythm++;if (rhythm % 150 == 0&&rhythm<=600&&ps->_SleepTime>=45){ps->_SleepTime -= 15;ps->_foodWeight += 2;}//重新刷新障碍物if (rhythm%200==0){IMAGE b;//设置背景loadimage(&b, "blackground.bmp", SIZE, SIZE, true);//释放之前的障碍物pSnakesNode tmp = ps->_pBarrier;while (tmp != NULL){putimage(tmp->x, tmp->y, &b);pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}ps->_pBarrier = NULL;barrier(ps);}} while (ps->_Status==OK);//判断游戏状态}
游戏结束
1、游戏结束后,通过判断游戏状态,来告诉玩家是什么原因无的。
2、释放蛇、食物、障碍物,最后结束程序。
/游戏结束
void GameEnd(pSnakes ps)
{
//判断是什么原因结束的if (ps->_Status == END_NOMAL){outtextxy(340, 300, "正常结束游戏");}else if (ps->_Status == KILL_BY_SELF){outtextxy(340, 300, "咬到自己了");}else if (ps->_Status == KILL_BY_WALL){outtextxy(340, 300, "撞墙了");}else if (ps->_Status == KILL_BY_BARR){outtextxy(340, 300, "撞到障碍物了");}//释放蛇pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}//释放障碍物tmp = ps->_pBarrier;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}//最后释放食物tmp = ps->_pFood;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}//按任意键system("pause");//结束程序exit(1);
}
总参考代码:
test.cpp
#include"Snakes.h"
void test()
{srand((unsigned int)time(NULL));Snakes snake = {0};GameStart(&snake);GameRun(&snake);GameEnd(&snake);
}
int main()
{test();return 0;
}
Snakes.h
#pragma once
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<Windows.h>
#include<locale.h>
#include<time.h>
#include<stdbool.h>
#include<assert.h>
#include<graphics.h>
#include <conio.h>
#include<string>//按键
#define KEY_PRESS(VK) ((GetAsyncKeyState(VK)&0x1) ? 1 : 0)//蛇的初始位置
#define POS_X 0
#define POS_Y 0
#define SIZE 20
//⽅向
enum DIRECTION
{UP = 1,DOWN,LEFT,RIGHT
};//游戏状态
enum GAME_STATUS
{OK,//正常运⾏KILL_BY_WALL,//撞墙KILL_BY_SELF,//咬到⾃⼰KILL_BY_BARR,//撞到障碍物了END_NOMAL//正常结束
};//蛇的节点和食物
typedef struct SnakesNode
{//坐标int x;int y;//链接struct SnakesNode* next;
}SnakesNode, *pSnakesNode;//维护整个运行
typedef struct Snakes
{pSnakesNode _pSnake;//维护整条蛇的指针pSnakesNode _pFood;//维护⻝物的指针pSnakesNode _pBarrier;//维护障碍物的指针enum DIRECTION _Dir;//蛇头的⽅向默认是向右enum GAME_STATUS _Status;//游戏状态int _Socre;//当前获得分数int _foodWeight;//默认每个⻝物10分int _SleepTime;//每⾛⼀步休眠时间
}Snakes,*pSnakes;//游戏开始前的初始化
void GameStart(pSnakes ps);//欢迎界⾯
//void WelcomeToGame();
//创建地图
void CreateMap();
//设置光标信息
void SetPos(short x, short y);
//初始化蛇
void InitSnake(pSnakes ps);
//生成障碍物void barrier(pSnakes ps);
//创建⻝物
void CreateFood(pSnakes ps);
// 暂停响应
void pause();
//下⼀个节点是⻝物
void EatFood(pSnakesNode psn, pSnakes ps);
//下一个节点不是食物
void NoFood(pSnakesNode psn, pSnakes ps);
//蛇的移动
void SnakeMove(pSnakes ps);
//撞墙检测
int KillByWall(pSnakes ps);
//撞⾃⾝检测
int KillBySelf(pSnakes ps);
//游戏运⾏
void GameRun(pSnakes ps);
//游戏结束
void GameEnd(pSnakes ps);
Snakes.cpp
#include"Snakes.h"//控制游戏节奏
int rhythm = 0;
//设置光标的坐标
void SetPos(short x, short y)
{COORD pos = { x, y };HANDLE hOutput = NULL;//获取标准输出的句柄(⽤来标识不同设备的数值)hOutput = GetStdHandle(STD_OUTPUT_HANDLE);//设置标准输出上光标的位置为posSetConsoleCursorPosition(hOutput, pos);
}//欢迎界⾯
void WelcomeToGame(IMAGE *bg_img)
{//在中间打印outtextxy(340, 300, "欢迎进入游戏...");//在末尾提示outtextxy(0, 620, "请按任意键开始游戏...");//任意键继续函数system("pause");//重新加载背景putimage(0, 0, bg_img);//游戏按键提示outtextxy(190, 300, "⽤ ↑ . ↓ . ← . → 分别控制蛇的移动, F3为加速,F4为减速");outtextxy(0, 620, "请按任意键开始游戏...");system("pause");//再重新加载背景putimage(0, 0, bg_img);}void Node(pSnakesNode p)
{IMAGE node;loadimage(&node, "屏幕截图 2024-01-28 140514.png", SIZE, SIZE, true);putimage(p->x, p->y, &node);
}
void Barr(pSnakesNode p)
{IMAGE barrier;loadimage(&barrier, "屏幕截图 2024-01-28 165936.png", SIZE, SIZE, true);putimage(p->x ,p->y, &barrier);
}
void Foob(pSnakesNode p)
{IMAGE food;loadimage(&food, "屏幕截图 2024-01-28 151002.png", SIZE, SIZE, true);putimage(p->x, p->y, &food);
}//创建单个食物
pSnakesNode Create(pSnakes ps)
{pSnakesNode p;int x = 0;int y = 0;
again:
in:do {//1、要在地图内x = rand() % 780 + 1;y = rand() % 620 + 1;//2、x需要在奇数上} while (x % SIZE != 0 || y % SIZE != 0);//3、不能出现在蛇的身上pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){if (tmp->x == x && tmp->y == y){goto again;}tmp = tmp->next;}//4、不能出现在障碍物身上pSnakesNode tmp1 = ps->_pBarrier;while (tmp1 != NULL){if (tmp1->x == x && tmp1->y == y){goto in;}tmp1 = tmp1->next;}p = (pSnakesNode)malloc(sizeof(SnakesNode));assert(p);p->next = NULL;p->x = x;p->y = y;Foob(p);return p;}//创建⻝物
void CreateFood(pSnakes ps)
{pSnakesNode p;int x = 0;int y = 0;for (int i = 0; i < 10; i++){pSnakesNode p = Create(ps);if (ps->_pFood == NULL){ps->_pFood= p;}else{p->next = ps->_pFood;ps->_pFood = p;p = NULL;}}
}
//初始化蛇身
void InitSnake(pSnakes ps)
{//头节点pSnakesNode tmp = (pSnakesNode)malloc(sizeof(SnakesNode));//判断是否成功assert(tmp);//初始化第一个节点tmp->next = NULL;tmp->x = POS_X;tmp->y = POS_Y;//将这个节点加载到设备上Node(tmp);//有头插法将其他4个节点加入for (int i = 1; i <=4; i++){pSnakesNode tmp1 = (pSnakesNode)malloc(sizeof(SnakesNode));assert(tmp1); tmp1->x = POS_X+i*SIZE;tmp1->y = POS_Y;//新的接后的一个tmp1->next = tmp;tmp = tmp1;tmp1 = NULL;Node(tmp);}//将结构体中维护蛇节点的指针赋值ps->_pSnake = tmp;tmp = NULL;//其他属性初始化ps->_Dir = RIGHT;ps->_Socre = 0;ps->_Status = OK;ps->_SleepTime = 150;ps->_foodWeight = 10;ps->_pBarrier = NULL;
}
//生成障碍物
void barrier(pSnakes ps)
{//障碍物的坐标int x = 0;int y = 0;pSnakesNode p;//生成障碍物的坐标for (int i = 0; i < 10; i++){ again://跳回do {//1、要在地图内x = rand() % 780 + 1;y = rand() % 620 + 1;//2、x和y要在SIZE的倍数上} while (x % SIZE != 0 || y % SIZE != 0);//3、不能出现在蛇的身上pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){if (tmp->x == x && tmp->y == y){//如果碰到是在蛇的身上的话就跳转到开始的位置goto again;}tmp = tmp->next;}p = (pSnakesNode)malloc(sizeof(SnakesNode));assert(p);p->next = NULL;p->x = x;p->y = y;//加载障碍物图片Barr(p);//如果ps->_pBarrier == NULL则将p赋给ps->_pBarrier,其他的用头插插入if (ps->_pBarrier == NULL){ps->_pBarrier = p;}else{p->next = ps->_pBarrier;ps->_pBarrier = p;p = NULL;} }
}void GameStart(pSnakes ps)
{//绘制窗口大小initgraph(1000, 640);//图像对象IMAGE bg_img;//设置背景loadimage(&bg_img, "blackground.bmp", 800, 640, true);//加载到设备putimage(0, 0, &bg_img);//欢迎界面WelcomeToGame(&bg_img);//创造地图//CreateMap();//蛇身初始化InitSnake(ps);//生成障碍物barrier(ps);//创建⻝物CreateFood(ps);
}
//打印右侧帮助信息
void PrintHelpInfo()
{//打印提⽰信息SetPos(64, 15);printf("不能穿墙,不能咬到⾃⼰\n");SetPos(64, 16);printf("⽤↑.↓.←.→分别控制蛇的移动.");SetPos(64, 17);printf("F1 为加速,F2 为减速\n");SetPos(64, 18);printf("ESC :退出游戏.space:暂停游戏.");
}
//暂停
void pause()
{while (1){Sleep(300);if (KEY_PRESS(VK_SPACE)){break;}}
}
//判断是不是食物
bool judgment(pSnakesNode psn, pSnakes ps)
{pSnakesNode tmp = ps->_pFood;while (tmp != NULL){if (tmp->x == psn->x && tmp->y == psn->y){return true;}tmp = tmp->next;}return false;
}
//下⼀个节点是⻝物
void EatFood(pSnakesNode psn, pSnakes ps)
{//直接将下一个坐标和当前的贪吃蛇链接,并加载图片psn->next = ps->_pSnake;ps->_pSnake = psn;Node(psn);//找食物节点//前pSnakesNode p = ps->_pFood;//后pSnakesNode p1 =NULL;while (p != NULL){if (p->x == psn->x && p->y == psn->y){break;}p1 = p;p = p->next;}//生成一个食物pSnakesNode p2 = Create(ps);//头if (p1 == NULL){p2->next = p->next;free(p);ps->_pFood = p2;}else{p1->next = p2;p2->next = p->next;free(p);}//加分ps->_Socre += ps->_foodWeight;
}//下一个节点不是食物
void NoFood(pSnakesNode psn, pSnakes ps)
{psn->next = ps->_pSnake;ps->_pSnake = psn;pSnakesNode tmp = ps->_pSnake;//找倒数第二个节点while (tmp->next->next!= NULL){Node(tmp);tmp = tmp->next;}IMAGE b;//设置背景loadimage(&b, "blackground.bmp", SIZE, SIZE, true);putimage(tmp->next->x, tmp->next->y, &b);//释放最后一个free(tmp->next);tmp->next = NULL;
}
//撞墙检测
int KillByWall(pSnakes ps)
{if (ps->_pSnake->x<0 || ps->_pSnake->x>780 || ps->_pSnake->y<0|| ps->_pSnake->y>620){ps->_Status = KILL_BY_WALL;return 1;}else{return 0;}
}
//撞⾃⾝检测
int KillBySelf(pSnakes ps)
{pSnakesNode tmp = ps->_pSnake->next;while (tmp){if (tmp->x == ps->_pSnake->x && tmp->y == ps->_pSnake->y){ps->_Status= KILL_BY_SELF;return 1;}tmp = tmp->next;}return 0;
}
//撞到障碍物
int Hittheobstacles(pSnakes ps)
{pSnakesNode tmp = ps->_pSnake;pSnakesNode tmp1 = ps->_pBarrier;while (tmp1 != NULL){if (tmp1->x == tmp->x && tmp1->y == tmp->y){ps->_Status = KILL_BY_BARR;return 1;}tmp1 = tmp1->next;}return 0;
}//蛇的移动
void SnakeMove(pSnakes ps)
{//计算下一个坐标的位置// //下一个节点pSnakesNode tmp = (pSnakesNode)malloc(sizeof(SnakesNode));assert(tmp);tmp->next=NULL;//上if (ps->_Dir == UP){tmp->y = ps->_pSnake->y - SIZE;tmp->x = ps->_pSnake->x;}//下else if (ps->_Dir == DOWN){tmp->y = ps->_pSnake->y + SIZE;tmp->x = ps->_pSnake->x;}//左else if (ps->_Dir == LEFT){tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x-SIZE;}//右else if (ps->_Dir == RIGHT){tmp->y = ps->_pSnake->y;tmp->x = ps->_pSnake->x+SIZE;}//判断是否是食物//是食物if (judgment(tmp,ps)){ EatFood(tmp, ps);}//不是食物else{NoFood(tmp, ps);}KillByWall(ps);KillBySelf(ps);Hittheobstacles(ps);
}//游戏运⾏
void GameRun(pSnakes ps)
{do {//记录的分char str[10] = { 0 };//利用该函数将当前得分输入到str上sprintf(str, "%d", ps->_Socre);//记录当前食物的分数char str1[5] = { 0 };sprintf(str1, "%d", ps->_foodWeight);//在规定的位置上输出outtextxy(810, 20, "当前得分:");outtextxy(940, 20, str);outtextxy(810, 80, "当前食物得分:");outtextxy(940, 80, str1);if (KEY_PRESS(VK_UP) && ps->_Dir != DOWN){ps->_Dir = UP;}else if (KEY_PRESS(VK_DOWN) && ps->_Dir != UP){ps->_Dir = DOWN;}else if (KEY_PRESS(VK_LEFT) && ps->_Dir != RIGHT){ps->_Dir = LEFT;}else if (KEY_PRESS(VK_RIGHT) && ps->_Dir != LEFT){ps->_Dir = RIGHT;}else if (KEY_PRESS(VK_SPACE)){pause();}else if (KEY_PRESS(VK_ESCAPE)){ps->_Status = END_NOMAL;break;}else if (KEY_PRESS(VK_F3)){if (ps->_SleepTime >= 50){ps->_SleepTime -= 20;ps->_foodWeight += 2;}}else if (KEY_PRESS(VK_F4)){if (ps->_SleepTime < 350){ps->_SleepTime += 20;ps->_foodWeight -= 2;if (ps->_SleepTime == 350){ps->_foodWeight = 1;}}}//暂停Sleep(ps->_SleepTime);//移动蛇SnakeMove(ps);rhythm++;if (rhythm % 150 == 0&&rhythm<=600&&ps->_SleepTime>=45){ps->_SleepTime -= 15;ps->_foodWeight += 2;}if (rhythm%200==0){IMAGE b;//设置背景loadimage(&b, "blackground.bmp", SIZE, SIZE, true);//释放pSnakesNode tmp = ps->_pBarrier;while (tmp != NULL){putimage(tmp->x, tmp->y, &b);pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}ps->_pBarrier = NULL;barrier(ps);}} while (ps->_Status==OK);}
//游戏结束
void GameEnd(pSnakes ps)
{if (ps->_Status == END_NOMAL){outtextxy(340, 300, "正常结束游戏");}else if (ps->_Status == KILL_BY_SELF){outtextxy(340, 300, "咬到自己了");}else if (ps->_Status == KILL_BY_WALL){outtextxy(340, 300, "撞墙了");}else if (ps->_Status == KILL_BY_BARR){outtextxy(340, 300, "撞到障碍物了");}//释放蛇pSnakesNode tmp = ps->_pSnake;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}//释放障碍物tmp = ps->_pBarrier;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}//最后释放食物tmp = ps->_pFood;while (tmp != NULL){pSnakesNode tmp1 = tmp->next;free(tmp);tmp = tmp1;}system("pause");exit(1);
}
以上就是我的分享了,如果有什么错误,欢迎在评论区留言。
最后,祝大家天天开心
相关文章:

c语言实战之贪吃蛇
文章目录 前言效果展示游戏用到的图片游戏思路一览游戏前准备一、贪吃蛇、食物、障碍物节点坐标的结构体二、枚举游戏状态、和贪吃蛇的方向三、维护运行的结构体 游戏开始前的初始化一、学习图形库相关知识二、设置背景三、欢迎界面四、初始化贪吃蛇五、生成障碍物六、生成食物…...
Midjourney图片生成描述词记录(今天一天)
抄别人的描述词 /imagine prompt:https://(你的图片地址).jpg Super handsome boy IP by pop mart , green suit, no hair, bald head, Scenes in spring , pastel color , mockup , fine luster , clean background ,3D render , Soft focus , oc , bl…...

类和对象 第五部分第四小节:赋值运算符重载
C编译器至少给一个类添加4个函数 1.默认构造函数无参,函数体为空 2.默认析构函数无参,函数体为空 3.默认拷贝沟早函数,对属性进行值拷贝 4.赋值运算符“operator”,对属性进行值拷贝 如果类中有属性指向堆区,做赋值操作…...

Django从入门到精通(一)
目录 一、Django环境搭建与命令 1.1、安装 1.2、命令行 创建项目 编写代码 运行 app概念 1.3、Pycharm创建项目 1.4、虚拟环境 创建虚拟环境 - 命令行 介绍 操作 基本问题 Pycharm 项目虚拟环境 django虚拟环境【安装django最新版本】 django虚拟环境【安装指…...
数据库分表分库的原则
什么是数据库分库分表 数据库分表(Table Sharding) 数据库分表是将一个大表按照某种规则拆分成多个小表存储在不同的物理表中的技术。通常,拆分规则是基于某个列的值进行拆分,例如根据用户ID或日期范围等进行拆分。每个小表只包…...

Java技术栈 —— Docker容器
Java技术栈 —— Docker容器 一、什么是Docker?二、如何安装Docker?三、如何使用Docker? 一、什么是Docker? docker的本意是码头工人。 Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个…...

Mysql大数据量分页优化
前言 之前有看过到mysql大数据量分页情况下性能会很差,但是没有探究过它的原因,今天讲一讲mysql大数据量下偏移量很大,性能很差的问题,并附上解决方式。 原因 将原因前我们先做一个试验,我做试验使用的是mysql5.7.2…...

QT tcp与udp网络通信以及定时器的使用 (7)
QT tcp与udp网络通信以及定时器的使用 文章目录 QT tcp与udp网络通信以及定时器的使用1、QT网络与通信简单介绍2、QT TCP通信1、 服务器的流程2、 客户端的流程3、服务器的编写4、客户端的编写 3、QT UDP通信1、客户端流程2、客户端编写3、UDP广播4、UDP组播 4、定时器的用法1、…...
web架构师编辑器内容-添加自动保存的功能
对于频繁改动的应用,自动保存的功能是一个非常有用的功能,可以避免用户在没有保存的情况下丢失自己保存过的数据。 对于自动保存,一般有两种实现,参考语雀和石墨: 语雀采用的是定时保存的方式,大约在3分半…...

【Redis】关于它为什么快?使用场景?以及使用方式?为何引入多线程?
目录 1.既然redis那么快,为什么不用它做主数据库,只用它做缓存? 2.Redis 一般在什么场合下使用? 3.redis为什么这么快? 4.Redis为什么要引入了多线程? 1.既然redis那么快,为什么不用它做主数据…...

SpringBoot之JWT登录
JWT JSON Web Token(JSON Web令牌) 是一个开放标准(rfc7519),它定义了一种紧凑的、自包含的方式,用于在各方之间以JSON对象安全地传输信息。此信息可以验证和信任,因为它是数字签名的。jwt可以使用秘密〈使用HNAC算法…...

【备战蓝桥杯】——循环结构
🌈个人主页: Aileen_0v0 🔥热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法 💫个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-bFHV3Dz5xMe6d3NB {font-family:"trebuchet ms",verdana,arial,sans-serif;font-siz…...

【数据分享】1929-2023年全球站点的逐年平均气温数据(Shp\Excel\免费获取)
气象数据是在各项研究中都经常使用的数据,气象指标包括气温、风速、降水、湿度等指标,其中又以气温指标最为常用!说到气温数据,最详细的气温数据是具体到气象监测站点的气温数据!本次我们为大家带来的就是具体到气象监…...

探索Pyecharts关系图绘制技巧:炫酷效果与创意呈现【第42篇—python:Pyecharts水球图】
文章目录 Pyecharts绘制多种炫酷关系网图引言准备工作代码实战1. 基本关系网图2. 自定义节点样式和边样式3. 关系网图的层级结构4. 添加标签和工具提示5. 动态关系网图6. 高级关系网图 - Les Miserables 示例7. 自定义关系网图布局8. 添加背景图9. 3D 关系网图10. 热力关系网图…...

蓝桥杯-循环节长度
两个整数做除法,有时会产生循环小数,其循环部分称为: 循环节。比如,11/136>0.8461553846153..... 其循环节为[846153] 共有 6 位。下面的方法,可以求出循环节的长度。请仔细阅读代码,并填写划线部分缺少的代码。 注…...
Jython调用openwire库连接activemq转发topic订阅消息到另一个activemq 服务器上 完整代码
以下是一个示例代码,演示如何在Jython中使用OpenWire库连接ActiveMQ,将一个主题(topic)上的订阅消息转发到另一个ActiveMQ服务器上: from org.apache.activemq import * from org.apache.activemq.transport import *…...
面试经典题---30.串联所有单词的子串
30.串联所有单词的子串 我的解法: 滑动窗口: 解法中用到了两个哈希表map1和map2,分别用于记录words中各个单词的出现频数和当前滑动窗口[left, right)中单词的出现频数;外部for循环i从0到len - 1,内部while循环每次会…...
字符串随机生成工具(开源)-Kimen(奇门)
由于最近笔者在开发数据脱敏相关功能,其中一类脱敏需求为能够按照指定的格式随机生成一个字符串来代替原有信息,数据看起来格式需要与原数据相同,如:电话号码,身份证号以及邮箱等。在网上搜索了下,发现没有…...

UE4 CustomDepthMobile流程小记
原生UE opaque材质中获取CustomDepth/CustomStencil会报错 在其Compile中调用的函数中没有看到报错逻辑 材质节点的逻辑都没有什么问题,所以看一下报错 在HLSLMaterialTranslator::Translate中 修改之后 mobile流程的不透明材质可以直接获取SceneTexture::customd…...

Docker 基础篇
目录 一、Docker 简介 1. Docker 2. Linux 容器 3. 传统虚拟机和容器的对比 4. Docker 的作用 5. Docker 的基本组成(Docker 三要素) 6. Docker 工作原理 7. Docker 架构 8. Docker 下载 二、Docker 安装 1. CentOS Docker 安装 2. CentOS8 …...

IDEA运行Tomcat出现乱码问题解决汇总
最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明
LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造,完美适配AGV和无人叉车。同时,集成以太网与语音合成技术,为各类高级系统(如MES、调度系统、库位管理、立库等)提供高效便捷的语音交互体验。 L…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...

C++_核心编程_多态案例二-制作饮品
#include <iostream> #include <string> using namespace std;/*制作饮品的大致流程为:煮水 - 冲泡 - 倒入杯中 - 加入辅料 利用多态技术实现本案例,提供抽象制作饮品基类,提供子类制作咖啡和茶叶*//*基类*/ class AbstractDr…...

大话软工笔记—需求分析概述
需求分析,就是要对需求调研收集到的资料信息逐个地进行拆分、研究,从大量的不确定“需求”中确定出哪些需求最终要转换为确定的“功能需求”。 需求分析的作用非常重要,后续设计的依据主要来自于需求分析的成果,包括: 项目的目的…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...

使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...

CentOS下的分布式内存计算Spark环境部署
一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架,相比 MapReduce 具有以下核心优势: 内存计算:数据可常驻内存,迭代计算性能提升 10-100 倍(文档段落:3-79…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力
引言: 在人工智能快速发展的浪潮中,快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型(LLM)。该模型代表着该领域的重大突破,通过独特方式融合思考与非思考…...