百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

C语言实现贪吃蛇小游戏!(超简单详细)详细思路+源码分享

ztj100 2024-10-29 18:21 15 浏览 0 评论

贪吃蛇(也叫做贪食蛇)游戏是一款休闲益智类游戏,有PC和手机等多平台版本。既简单又耐玩。该游戏通过控制蛇头方向吃蛋,从而使得蛇变得越来越长。

我们的今天的目标就是:用C语言来实现一个贪吃蛇项目,也不用太复杂,初学者能看懂就行!

贪吃蛇小程序,主要采用C语言描述,采用数组进行数据存储的图形化操作界面,会涉及一点点C++ 知识(特别少的一点点),但是有C语言基础就够用了,编译器:VS2019,会涉及部分库函数的安装及使用,不过也很简单啦!

效果展示:

看到了这个效果之后,那我们就动手来实现吧!

希望大家都能轻松实现这个项目!另外涉及到的#include <graphics.h>需要额外安装图形库插件easyX哦~

代码如下:

#include <graphics.h>
#include <stdlib.h>
#include <conio.h>
#include <time.h>
#include <stdio.h>
#include "snake.h"
 
int score = 0;
int gamespeed = 100;                //蛇运行速度
 
static void Init(void);             /*图形驱动*/
static void Close(void);            /*图形结束*/
static void Game_interface(void);   /*游戏界面*/
static void GameOver(void);         /*结束游戏*/
static void GamePlay(void);         /*游戏过程*/
static void PrScore(void);          /*输出成绩*/
 
/*主函数*/
int main(void)
{
     Init();                 
     Game_interface();                 
     GamePlay();              
     Close();                 
     return 0;
}
 
/*图形驱动*/
static void Init(void)
{
    int gd=9,gm=2;
 
    initgraph(&gd,&gm," ");
    cleardevice();
}
 
/* 开始画面,左上角坐标为(50,40),右下角坐标为(610,460)的围墙 */
static void Game_interface(void)
{
     int i;
 
     setcolor(LIGHTCYAN);                           /*setbkcolor(LIGHTGREEN);*/
     setlinestyle(PS_SOLID,0,1);                    /*设置线型*/
     for(i=50;i<=600;i+=10)                          /*画边框*/
     {
         rectangle(i,40,i+10,49);                   /*上边框*/
         rectangle(i,451,i+10,460);                 /*下边框*/
     }
     for(i=40;i<=450;i+=10)
     {
         rectangle(50,i,59,i+10);                   /*左边框*/
         rectangle(601,i,610,i+10);                 /*右边框*/
     }
}
 
/* 游戏主函数 */
static void GamePlay(void)
{
     int i;
      
     srand(time(NULL));                             /*随机数发生器*/
     food.yes = 1;                                  /*1表示需要出现新食物,0表示已经存在食物*/
     snake.life = 0;                                /*活着*/
     snake.direction = 1;                           /*方向往右*/
     snake.x[0] = 100;
     snake.y[0] = 100;                              
     snake.x[1] = 110;
     snake.y[1] = 100;
     snake.node = 2;                                /*节数*/
 
     PrScore();                                     /*输出得分*/
     while(1)                                       /*可以重复玩游戏,压ESC键结束*/
     {
         while( !kbhit() )                          /*在没有按键的情况下,蛇自己移动*/
         {
             if(food.yes == 1)                      /*需要出现新食物*/
             {
                 food.x = rand()%400 + 60;
                 food.y = rand()%350 + 60;
                 while(food.x%10 != 0)              /*食物随机出现后必须让食物能够在整格内,这样才可以让蛇吃到*/
                     food.x++;
                 while(food.y%10 != 0)
                     food.y++;
                 food.yes = 0;                      /*画面上有食物了*/
             }
             if(food.yes == 0)                      /*画面上有食物了就要显示*/
             {
                 setcolor(GREEN);
                 rectangle(food.x,food.y,food.x + 10,food.y - 10);
             }
 
 
             for(i=snake.node-1;i>0;i--)          /*蛇的每个环节往前移动,也就是贪吃蛇的关键算法*/
             {
                 snake.x[i] = snake.x[i-1];
                 snake.y[i] = snake.y[i-1];
             }
             /*1,2,3,4表示右,左,上,下四个方向,通过这个判断来移动蛇头*/
             switch(snake.direction)
             {
                 case 1: 
                     snake.x[0] += 10;
                     break;
                 case 2: 
                     snake.x[0] -= 10;
                     break;
                 case 3: 
                     snake.y[0] -= 10;
                     break;
                 case 4: 
                     snake.y[0] += 10;
                     break;
             }
             /* 从蛇的第四节开始判断是否撞到自己 */
             for(i=3;i<snake.node;i++)
             {
                 if((snake.x[i] == snake.x[0]) && (snake.y[i] == snake.y[0]))
                 {
                     GameOver();                  /*显示失败*/
                     snake.life = 1;
                     break;
                 }
             }
             if((snake.x[0] < 55) || (snake.x[0] > 595) || (snake.y[0] < 55) || (snake.y[0] > 455))     /*蛇是否撞到墙壁*/
             {
 
 
                 GameOver();                                            /*本次游戏结束*/
                 snake.life = 1;                                        /*蛇死*/
             }
             if(snake.life == 1)                                        /*以上两种判断以后,如果蛇死就跳出内循环,重新开始*/
                 break;
             if((snake.x[0] == food.x) && (snake.y[0] == food.y))       /*吃到食物以后*/
             {
                 setcolor(BLACK);                                       /*把画面上的食物东西去掉*/
                 rectangle(food.x,food.y,food.x+10,food.y-10);
                 snake.x[snake.node] = -20;
                 snake.y[snake.node] = -20;
 
                 /* 新的一节先放在看不见的位置,下次循环就取前一节的位置 */
 
                 snake.node++;                      /*蛇的身体长一节*/
                 food.yes = 1;                      /*画面上需要出现新的食物*/
                 score += 10;
                 PrScore();                         /*输出新得分*/
             }
             setcolor(RED);                         /*画出蛇*/
 
 
             for(i=0;i<snake.node;i++)
                 rectangle(snake.x[i],snake.y[i],snake.x[i]+10,snake.y[i]-10);
             Sleep(gamespeed);
             setcolor(BLACK);                        /*用黑色去除蛇的的最后一节*/
             rectangle(snake.x[snake.node-1],snake.y[snake.node-1],
                       snake.x[snake.node-1]+10,snake.y[snake.node-1]-10);
         }        /*endwhile(!kbhit)*/
          
         if(snake.life == 1)                        /*如果蛇死就跳出循环*/
             break;
          
         key=getch();                          /*接收按键*/
         if (key == ESC) break;                  /*按ESC键退出*/
          
         switch(key)
         {                                 
             case UP:
                 if(snake.direction != 4)           /*判断是否往相反的方向移动*/
                     snake.direction = 3;
                 break;
             case RIGHT:
                 if(snake.direction != 2)
                     snake.direction = 1;
                 break;
             case LEFT:
                 if(snake.direction != 1)
                     snake.direction = 2;
                 break;
             case DOWN:
                 if(snake.direction != 3)
                     snake.direction = 4;
                 break;
         }
 
     }/*endwhile(1)*/
}
 
/*游戏结束*/
static void GameOver(void)
{
     cleardevice();
     PrScore();
     setcolor(RED);
     setfont(56,0,"黑体");
     outtextxy(200,200,"GAME OVER");
     getch();
}
 
/*输出成绩*/
static void PrScore(void)
{
     char str[10];
 
     setfillstyle(YELLOW);
     bar(50,15,220,35);
     setcolor(BROWN);
     setfont(16,0,"宋体");
     sprintf(str,"score:%d",score);
     outtextxy(55,16,str);
}
 
static void Close(void)
{
     closegraph();
}
?
#ifndef SNAKE_H
#define SNAKE_H
 
 
#define LEFT 'a'
#define RIGHT 'd'
#define DOWN 's'
#define UP 'w'
#define ESC 27
 
#define N 200               /*蛇的最大长度*/
 
char key;                   /*控制按键*/
 
 
struct Food
{
     int x;                   /*食物的横坐标*/
     int y;                   /*食物的纵坐标*/
     int yes;                 /*判断是否要出现食物的变量*/
}food;                        /*食物的结构体*/
 
struct Snake
{
     int x[N];
     int y[N];
     int node;                /*蛇的节数*/
     int direction;           /*蛇移动方向*/
     int life;                /* 蛇的生命,0活着,1死亡*/
}snake;
 
 
 
#endif?

此外,我也给大家分享我收集的其他资源,从最零基础开始的教程到C语言C++项目案例,帮助大家在学习C语言的道路上披荆斩棘!

编程学习书籍分享:

编程学习视频分享:

整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)最重要的是你可以在群里面交流提问编程问题哦!

对于C/C++感兴趣可以关注小编在后台私信我:【编程交流】一起来学习哦!可以领取一些C/C++的项目学习视频资料哦!已经设置好了关键词自动回复,自动领取就好了!

相关推荐

如何将数据仓库迁移到阿里云 AnalyticDB for PostgreSQL

阿里云AnalyticDBforPostgreSQL(以下简称ADBPG,即原HybridDBforPostgreSQL)为基于PostgreSQL内核的MPP架构的实时数据仓库服务,可以...

Python数据分析:探索性分析

写在前面如果你忘记了前面的文章,可以看看加深印象:Python数据处理...

CSP-J/S冲奖第21天:插入排序

...

C++基础语法梳理:算法丨十大排序算法(二)

本期是C++基础语法分享的第十六节,今天给大家来梳理一下十大排序算法后五个!归并排序...

C 语言的标准库有哪些

C语言的标准库并不是一个单一的实体,而是由一系列头文件(headerfiles)组成的集合。每个头文件声明了一组相关的函数、宏、类型和常量。程序员通过在代码中使用#include<...

[深度学习] ncnn安装和调用基础教程

1介绍ncnn是腾讯开发的一个为手机端极致优化的高性能神经网络前向计算框架,无第三方依赖,跨平台,但是通常都需要protobuf和opencv。ncnn目前已在腾讯多款应用中使用,如QQ,Qzon...

用rust实现经典的冒泡排序和快速排序

1.假设待排序数组如下letmutarr=[5,3,8,4,2,7,1];...

ncnn+PPYOLOv2首次结合!全网最详细代码解读来了

编辑:好困LRS【新智元导读】今天给大家安利一个宝藏仓库miemiedetection,该仓库集合了PPYOLO、PPYOLOv2、PPYOLOE三个算法pytorch实现三合一,其中的PPYOL...

C++特性使用建议

1.引用参数使用引用替代指针且所有不变的引用参数必须加上const。在C语言中,如果函数需要修改变量的值,参数必须为指针,如...

Qt4/5升级到Qt6吐血经验总结V202308

00:直观总结增加了很多轮子,同时原有模块拆分的也更细致,估计为了方便拓展个管理。把一些过度封装的东西移除了(比如同样的功能有多个函数),保证了只有一个函数执行该功能。把一些Qt5中兼容Qt4的方法废...

到底什么是C++11新特性,请看下文

C++11是一个比较大的更新,引入了很多新特性,以下是对这些特性的详细解释,帮助您快速理解C++11的内容1.自动类型推导(auto和decltype)...

掌握C++11这些特性,代码简洁性、安全性和性能轻松跃升!

C++11(又称C++0x)是C++编程语言的一次重大更新,引入了许多新特性,显著提升了代码简洁性、安全性和性能。以下是主要特性的分类介绍及示例:一、核心语言特性1.自动类型推导(auto)编译器自...

经典算法——凸包算法

凸包算法(ConvexHull)一、概念与问题描述凸包是指在平面上给定一组点,找到包含这些点的最小面积或最小周长的凸多边形。这个多边形没有任何内凹部分,即从一个多边形内的任意一点画一条线到多边形边界...

一起学习c++11——c++11中的新增的容器

c++11新增的容器1:array当时的初衷是希望提供一个在栈上分配的,定长数组,而且可以使用stl中的模板算法。array的用法如下:#include<string>#includ...

C++ 编程中的一些最佳实践

1.遵循代码简洁原则尽量避免冗余代码,通过模块化设计、清晰的命名和良好的结构,让代码更易于阅读和维护...

取消回复欢迎 发表评论: