c语言编写的小游戏有哪些_2048游戏c语言程序设计

《2048》是最近比较流行的一款数字游戏。原版2048首先在github上发布,原作者是Gabriele Cirulli。它是基于《1024》和《小3传奇》(Threes!)的玩法开发而成的新型数字游戏。

c语言编写的小游戏有哪些_2048游戏c语言程序设计

游戏规则

游戏的规则很简单,你需要控制所有方块向同一个方向运动,两个相同数字的方块撞在一起之后合并成为他们的和,每次操作之后会在空白的方格处随机生成一个2或者4(生成2的概率要大一些),最终得到一个“2048”的方块就算胜利了。

核心算法

1、方块移动和合并算法

主要思想:把游戏数字面板抽象成4行4列的二维数组a,值为0的位置表示空方块,其他位置表示对应数字方块。把每一行同等对待,只研究一行的移动合并算法,然后可以通过遍历行来实现所有行的移动合并算法。在一行中,用b表示一行的一位数组,使用两个下标变量来遍历列项,这里使用j和k,其中j总在k的后面,用来寻找k项后面第一个不为0的数字,而k项用于表示当前待比较的项,总是和j项之间隔着若干个数字0,或者干脆紧挨着。不失一般性,考虑往左滑动时,初始情况下j等于1,而k等于0,接着判断j项数字是否大于0,若是,则判断j项和k项数字的关系,分成3种情况处理,分别是P1: ,P2: b==0和P3: b!=0且b!=b;若否,则j自加1,然后继续寻找k项后面第一个不为0的数字。其中P1,P2和P3分别对应如下:

P1:b==b,则b = 2 * b(说明两数合并了),且b = 0(合并之后要将残留的j项值清零),接着k自加1,然后进行下一次循环。

P2:b==0,则表示b之前全是空格子,此时直接移动b到k的位置,也就是b = b,然后b = 0(移动后将残留的j项值清零),接着k值不变,然后进行下一次循环。

P3:b!=0且b!=b,则表示两数不相等且都不为0,此时将两数靠在一起,也就是b[k+1] = b。接着分两种小情况,如j!=k+1,则b = 0(移动后将残留的j项值清零);若否,则表示两数原先就靠在一起,则不进行特殊处理(相当于未移动)。接着k字加1,然后进行下一次循环。

2、判断游戏是否结束算法

核心思想:遍历二维数组,看是否存在横向和纵向两个相邻的元素相等,若存在,则游戏结束,若不存在,则游戏结束。

3、生成随机数算法

核心思想:根据生成的随机数,对一定的值进行取模,达到生成一定概率的数。在本游戏中,设定出现2的概率是4的两倍,于是可以利用系统提供的随机数函数生成一个数,然后对3区域,得到的数若小于2则在游戏面板空格处生成一个2,若余数等于2,则生成4。在选择将在哪一个空格出生成数的时候,也是根据系统提供的随机函数生成一个数,然后对空格数取余,然后在第余数个空格出生成数字。

4、绘制界面的算法

核心思想:利用系统提供的控制台界面清屏功能,达到刷新界面的效果,利用控制制表符位置,达到绘制游戏数字面板的效果。

由于绘制界面不算是本游戏的本质,且代码段相对较长,所以算法描述在这里省略,读者可以参考完整源代码。

源码示例:

#include <stdio.h> #include <time.h> /* 包含设定随机数种子所需要的time()函数 */ #include <conio.h> /* 包含Windows平台上完成输入字符不带回显和回车确认的getch()函数 */ #include <windows.h> /* 包含Windows平台上完成设定输出光标位置达到清屏功能的函数 */ void start_game(); /* 开始游戏 */ void reset_game(); /* 重置游戏 */ /* 往左右上下四个方向移动 */ void move_left(); void move_right(); void move_up(); void move_down(); void refresh_show(); /* 刷新界面显示 */ void add_rand_num(); /* 生成随机数,本程序中仅生成2或4,概率之比设为2:1 */ void check_game_over(); /* 检测是否输掉游戏,设定游戏结束标志 */ int get_null_count(); /* 获取游戏面板上空位置数量 */ int board; /* 游戏数字面板,抽象为二维数组 */ int score; /* 游戏的分 */ int best; /* 游戏最高分 */ int if_need_add_num; /* 是否需要生成随机数标志,1表示需要,0表示不需要 */ int if_game_over; /* 是否游戏结束标志,1表示游戏结束,0表示正常 */ /* main函数 函数定义 */ int main() { start_game(); } /* 开始游戏 函数定义 */ void start_game() { reset_game(); char cmd; while (1) { cmd = getch(); /* 接收标准输入流字符命令 */ if (if_game_over) /* 判断是否已经输掉游戏 */ { if (cmd == 'y' || cmd == 'Y') /* 重玩游戏 */ { reset_game(); continue; } else if (cmd == 'n' || cmd == 'N') /* 退出 */ { return; } else { continue; } } if_need_add_num = 0; /* 先设定默认需要生成随机数,需要时再设定为1 */ switch (cmd) /* 命令解析,w,s,a,d字符代表上下左右命令 */ { case 'a': case 'A': case 75 : move_left(); break; case 's': case 'S': case 80 : move_down(); break; case 'w': case 'W': case 72 : move_up(); break; case 'd': case 'D': case 77 : move_right(); break; } score > best ? best = score : 1; /* 打破得分纪录 */ if (if_need_add_num) /* 默认为需要生成随机数时也同时需要刷新显示,反之亦然 */ { add_rand_num(); refresh_show(); } } } /* 重置游戏 函数定义 */ void reset_game() { score = 0; if_need_add_num = 1; if_game_over = 0; /* 了解到游戏初始化时出现的两个数一定会有个2,所以先随机生成一个2,其他均为0 */ int n = rand() % 16; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { board = (n-- == 0 ? 2 : 0); } } /* 前面已经生成了一个2,这里再生成一个随机的2或者4,且设定生成2的概率是4的两倍 */ add_rand_num(); /* 在这里刷新界面并显示的时候,界面上已经默认出现了两个数字,其他的都为空(值为0) */ system("cls"); refresh_show(); } /* 生成随机数 函数定义 */ void add_rand_num() { srand(time(0)); int n = rand() % get_null_count();/* 确定在何处空位置生成随机数 */ for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { if (board == 0 && n-- == 0) /* 定位待生成的位置 */ { board = (rand() % 3 ? 2 : 4);/* 确定生成何值,设定生成2的概率是4的概率的两倍 */ return; } } } } /* 获取空位置数量 函数定义 */ int get_null_count() { int n = 0; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { board == 0 ? n++ : 1; } } return n; } /* 检查游戏是否结束 函数定义 */ void check_game_over() { for (int i = 0; i < 4; i++) { for (int j = 0; j < 3; j++) { /* 横向和纵向比较挨着的两个元素是否相等,若有相等则游戏不结束 */ if (board == board[j+1] || board == board[j+1]) { if_game_over = 0; return; } } } if_game_over = 1; } /* * 如下四个函数,实现上下左右移动时数字面板的变化算法 * 左和右移动的本质一样,区别仅仅是列项的遍历方向相反 * 上和下移动的本质一样,区别仅仅是行项的遍历方向相反 * 左和上移动的本质也一样,区别仅仅是遍历时行和列互换 */ /* 左一 函数定义 */ void move_left() { /* 变量i用来遍历行项的下标,并且在移动时所有行相互独立,互不影响 */ for (int i = 0; i < 4; i++) { /* 变量j为列下标,变量k为待比较(合并)项的下标,循环进入时k<j */ for (int j = 1, k = 0; j < 4; j++) { if (board > 0) /* 找出k后面第一个不为空的项,下标为j,之后分三种情况 */ { if (board == board) /* 情况1:k项和j项相等,此时合并方块并计分 */ { score += board[k++] <<= 1; board = 0; if_need_add_num = 1; /* 需要生成随机数和刷新界面 */ } else if (board == 0) /* 情况2:k项为空,则把j项赋值给k项,相当于j方块移动到k方块 */ { board = board; board = 0; if_need_add_num = 1; } else /* 情况3:k项不为空,且和j项不相等,此时把j项赋值给k+1项,相当于移动到k+1的位置 */ { board[++k] = board; if (j != k) /* 判断j项和k项是否原先就挨在一起,若不是则把j项赋值为空(值为0) */ { board = 0; if_need_add_num = 1; } } } } } } /* 右移 函数定义 */ void move_right() { /* 仿照左移操作,区别仅仅是j和k都反向遍历 */ for (int i = 0; i < 4; i++) { for (int j = 2, k = 3; j >= 0; j--) { if (board > 0) { if (board == board) { score += board[k--] <<= 1; board = 0; if_need_add_num = 1; } else if (board == 0) { board = board; board = 0; if_need_add_num = 1; } else { board[--k] = board; if (j != k) { board = 0; if_need_add_num = 1; } } } } } } /* 上衣 函数定义 */ void move_up() { /* 仿照左移操作,区别仅仅是行列互换后遍历 */ for (int i = 0; i < 4; i++) { for (int j = 1, k = 0; j < 4; j++) { if (board > 0) { if (board == board) { score += board[k++] <<= 1; board = 0; if_need_add_num = 1; } else if (board == 0) { board = board; board = 0; if_need_add_num = 1; } else { board[++k] = board; if (j != k) { board = 0; if_need_add_num = 1; } } } } } } /* 下移 函数定义 */ void move_down() { /* 仿照左移操作,区别仅仅是行列互换后遍历,且j和k都反向遍历 */ for (int i = 0; i < 4; i++) { for (int j = 2, k = 3; j >= 0; j--) { if (board > 0) { if (board == board) { score += board[k--] <<= 1; board = 0; if_need_add_num = 1; } else if (board == 0) { board = board; board = 0; if_need_add_num = 1; } else { board[--k] = board; if (j != k) { board = 0; if_need_add_num = 1; } } } } } } /* 刷新界面 函数定义 */ void refresh_show() { /* 重设光标输出位置方式清屏可以减少闪烁,system("cls")为备用清屏命令,均为Windows平台相关*/ COORD pos = {0, 0}; SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), pos); printf("\n\n\n\n"); printf(" GAME: 2048 SCORE: %06d BEST: %06d\n", score, best); printf(" --------------------------------------------------\n\n"); /* 绘制表格和数字 */ printf(" ┌──┬──┬──┬──┐\n"); for (int i = 0; i < 4; i++) { printf(" │"); for (int j = 0; j < 4; j++) { if (board != 0) { if (board < 10) { printf(" %d │", board); } else if (board < 100) { printf(" %d │", board); } else if (board < 1000) { printf(" %d│", board); } else if (board < 10000) { printf("%4d│", board); } else { int n = board; for (int k = 1; k < 20; k++) { n >>= 1; if (n == 1) { printf("2^%02d│", k); /* 超过四位的数字用2的幂形式表示,如2^13形式 */ break; } } } } else printf(" │"); } if (i < 3) { printf("\n ├──┼──┼──┼──┤\n"); } else { printf("\n └──┴──┴──┴──┘\n"); } } printf("\n"); printf(" --------------------------------------------------\n"); printf(" W↑ A← →D ↓S"); if (get_null_count() == 0) { check_game_over(); if (if_game_over) /* 判断是否输掉游戏 */ { printf("\r GAME OVER! TRY THE GAME AGAIN? [Y/N]"); } } }

效果示例:

c语言编写的小游戏有哪些_2048游戏c语言程序设计

希望大家能够很好地利用所学知识完成本项目!

写在最后:对于准备学习C/C++编程的小伙伴,如果你想更好的提升你的编程核心能力(内功)不妨从现在开始!

编程学习书籍分享:

c语言编写的小游戏有哪些_2048游戏c语言程序设计

编程学习视频分享:

c语言编写的小游戏有哪些_2048游戏c语言程序设计

整理分享(多年学习的源码、项目实战视频、项目笔记,基础入门教程)

欢迎转行和学习编程的伙伴,利用更多的资料学习成长比自己琢磨更快哦!

本文【c语言编写的小游戏有哪些_2048游戏c语言程序设计】由作者: 悲观锁 提供,本站不拥有所有权,只提供储存服务,如有侵权,联系删除!
本文链接:https://www.cuoshuo.com/blog/4291.html

(0)
上一篇 2023-03-13 09:02:09
下一篇 2023-03-08 10:35:57

相关推荐

  • css层叠样式表怎么做

    看完小编前段时间发表的HTML学习之后,相信大家对网站建设与网页编写有了一定的初步了解,今天我们来开始介绍层叠样式表css,css是为HTML页面的排版美化而设计的,可以实现html无法设计的效果,一般是用来配合html来使用的,一般模式:选择器{属性:值}即selector{property:value}例如:body{corlor:red;font-si…

    2023-03-13
    900
  • matlab递归函数怎么写 编写一个递归函数

    前言 本篇文章主要是记录一下在 GScript 中实现递归调用时所遇到的坑,类似的问题在中文互联网上我几乎没有找到相关的内容,所以还是很有必要记录一下。 在开始之前还是简单介绍下本次更新的 GScript v0.0.9 所包含的内容: 支持可变参数 优化 append 函数语义 优化编译错误信息 最后一个就是支持递归调用 先看第一个可变参数: //forma…

    2023-03-14
    200
  • 防止sql注入的方法有哪些_sql如何快速录入大量数据

    本文参考自owasp,重点是提供清晰,简单,可操作的指导,以防止应用程序中的SQL注入漏洞。不幸的是,SQL注入攻击很常见,这是由于两个因素: SQL注入漏洞的显着流行 目标的吸引力(即数据库通常包含应用程序的所有有趣/关键数据)。 发生了如此多的成功SQL注入攻击有点可耻,因为在代码中避免SQL注入漏洞非常简单。 当软件开发人员创建包含用户提供的输入的动态…

    2023-03-09
    900
  • until和not until的区别

    强调句的用法大家肯定已经很熟悉了,尤其是it做形式主语的强调句型。可是当句子里有not…until…的时候,很多朋友还是会搞混。今天我就跟大家一起挖一挖not…until…的强调句用法。 强调句的结构很简单,就是It is+被强调部分+that/who+…。比如陈述句:Tom lent me money when I was in trouble.我需要强调…

    2023-03-12
    600
  • 手机sd卡根目录是啥意思_手机sd卡在哪里找

    Sd卡就是我们所说的内存卡,它是用于存储较小的文件和数据的。存储卡通常用来扩展电子产品的容量,比如手机、相机等便携式电子产品。当存储卡安装在手机的时候,我们往往会访问它内部包含的各个目录和文件,那么sd的根目录和相关文件是如何存放的呢? 根目录的概念 根目录指逻辑驱动器的最上一级目录,它是相对于子目录来说的。比如电脑一般要分C盘,D盘等,我们点击C盘,进入的…

    2023-03-13
    600
  • 云计算应用场景有哪些

    本文来自35斗微信公众号,作者胡小凤。 随着大数据、人工智能技术的进步以及硬件设备的不断升级,农业发展正从“看天吃饭”到“知天而作”转变,自动化、智慧化的数字农业已成为未来农业的重要发展方向。 数字农业相关企业也成为资本争抢的香饽饽,据35斗不完全统计,2017-2022年我国数字农业领域投融资事件超百起,融资金额数十亿元。 不过,要成功实现农业数字化转型升…

    技术频道 2023-03-15
    200
  • 冒泡排序法和选择排序法的区别_c语言冒泡法排序10个数

    作者:绿皮长条瓜 目录 一、算法介绍 二、MATLAB实现 三、两个例子 一、算法介绍 冒泡排序(Bubble Sort)算法是一种计算科学领域的较简单的排序算法。它重复地走访过要排序的元素列,如果顺序(如从大到小、首字母从Z到A)错误就把他们交换过来。走访元素的工作是重复地进行直到没有相邻元素需要交换,也就是说该元素列已经排序完成。 算法步骤如下: ste…

    2023-03-11
    600
  • 计算机二级c语言题库及答案_c语言计算机二级题库

    1 选择题 1.1 C语言中,字符(char)型数据在计算机内存中的存储形式是: A) 反码 B) 补码 C) EBCDIC码 D) ASCII码 1.2 C语言中,下列不合法的字符常量是 A) '\xff' B) 'y=' C) '&' D) '\028' (8进制转义…

    2023-03-08
    600
  • python核心编程第二版

    人生苦短,我用Python!Python的学习对于数据分析是不可或缺的工具,虽然处理少量的数据时用Excel很便捷,但也一定不能忽视Python的学习,因为当涉及到大数据量、分维度的数据处理时,一定绕不开Python。即便是资深的数据分析师,也要时刻保持对Python领域专业知识的摄入,那有没有靠谱的Python数据分析书籍推荐呢?当然有。不仅有,还很多。以…

    2023-03-14
    100
  • continue语句只用于循环语句中吗_continue只能用于循环体中

    一般在switch循环遍历是经常用到break的,跳出当前循环。其实break也可以在for循环中使用,来结束当前循环,减少不必要的资源浪费; 网友一: 一、释义不同 1、break:间歇;休息;课间du休息;间断;zhi暂停。 2、continue:持续;继dao续存zhuan在;不断发生;继续做;不停shu地干;朝相同方向走,移动;延伸。 二、读音不同 …

    2023-03-10
    600

发表回复

登录后才能评论
返回顶部
错说博客上线啦!