用c语言编写2048小游戏_2048小游戏c语言课程设计

用c语言编写2048小游戏_2048小游戏c语言课程设计C语言实现简单的控制台2048小游戏。

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

一直很喜欢玩这个小游戏,简单的游戏中包含运气与思考与策略,喜欢这种简约又不失内涵的游戏风格。于是萌生了用C语言实现一下的想法。

具体代码是模仿这个:https://www.cnblogs.com/judgeyoung/p/3760515.html

 

博主分析的都很到位,很多算法技巧都值得借鉴,C语言实现2048的主要思想已经在那个博客中详细的分析了,但是我觉得在博主的代码中还是有很多很好的思想是值得我借鉴学习的。

比如这个生成随机数,顺便规定随机数的概率:

/* 生成随机数 函数定义 */
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[i][j] == 0 && n-- == 0) /* 定位待生成的位置 */
            {
                board[i][j] = (rand() % 3 ? 2 : 4);/* 确定生成何值,设定生成2的概率是4的概率的两倍 */
                return;
            }
        }
    }
}

 首先是 srand() 函数,他是一个随机数发生器的初始化函数。

原型为:void srand(unsigned seed)

用法是:程序员需要为这个函数提供一个随机数的种子:srand(随机数),如果使用相同的种子,那么后面的rand()函数就会每次运行都是生成一样的随机数,即伪随机数。

如:srand(1),直接用1来初始化种子,后面都是一样的随机数 。

为了生成真正的随机数,我们一般采用系统时间来作为随机数初始化函数的种子。使用time()函数来获取系统时间:

它的返回值为从 00:00:00 GMT, January 1, 1970 到现在所持续的秒数,然后将time_t型数据转化为(unsigned)型再传给srand函数,即: srand((unsigned) time(&t));

还有一个经常用法,不需要定义 time_t 型 t 变量,即: srand((unsigned) time(NULL)); 直接传入一个空指针,因为你的程序中往往并不需要经过参数获得的 t 数据。

第二句是:

int n = rand() % get_null_count();//在空余格中生成一个随机位置

利用随机数对剩余空格数目进行取余运算,得到小于剩余格数的随机数。

最后一句是:

if (board[i][j] == 0 && n-- == 0)//随机位置处为0时填入一个随机数,但是如果随机位置处不为0呢?
            {
                board[i][j] = (rand() % 3 ? 2 : 4);//在随机生成的空白格处填上一个2或者4,利用三项表达式对3取余,得到1/3,2/3的概率。
                return;//结束函数
            }

就是在格子中没有数字时并且在刚刚生成的随机数的位置处,填入一个数字2或者4,并且为了降低难度,固定生成2是生成4的概率的2倍。

利用三目运算符和对3取余的特点,产生概率分布。

rand()随机数对3取余只有可能是0/1/2,而在三目运算符中,当第一个数不为0时,运算符的值就取中间那个数的值,否则取最后一个数。所以取2的可能性为2/3,取4的可能性为1/3,这样就产生了不同的概率。但是这种方法只能产生  n:1 的概率分布,如果要产生      4:5,    8:17  的概率时,这种方法就不在适用了。

上面这些代码还是有一些漏洞的,因为游戏刚开始是需要有两个数的,一个数必为2,另一个数就是上面生成的那个数,但是如果第二个数恰好生成的随机位置处是第一个数,那么根据这儿代码,就什么也没有执行,最终导致出现刚开始界面就一个数字的情况。

解决办法就是在刚开始生成的两个数的程序中不使用  get_null_count(),而是获取第一个数字2的准确位置,然后生成第二个数字时,在排除第一个数字的地方生成数字即可。

程序的主体就是数字的上下左右移动,go_left()和其他三个函数,他们的思想都是相似的,就拿go_left()函数来分析一下:

按照原作者的思想,移动的时候一共有三种情况。

用c语言编写2048小游戏_2048小游戏c语言课程设计

如果相邻的两个数一样,就合并,数字相加。

如果相邻的数字不一样,简单的说,又分两种情况:数字需要移动和不需要移动。

具体代码为:

/*左移函数*/
void go_left(void)
{
    /*i遍历行下标*/
    for (int i = 0; i < 4; i++)
    {
        /*j为列下标,k为待比较项列下标,循环进入时k < j*/
        for (int j = 1,k = 0; j < 4; j++)
        {
            /*找出k后面第一个不为0的项*/
            if (board[i][j] > 0)
            {
                /*情况1*/
                if (board[i][j] == board[i][k])        //两个数相同就合并
                {
                    scoer += board[i][k++] <<= 1;
                    board[i][j] = 0;
                    if_need_add_num = 1;            //合并之后需要生成随机数和刷新界面
                }
                /*情况2*/
                else if (board[i][k] == 0)            //k项为空,则把j格移到k格
                {
                    board[i][k] = board[i][j];
                    board[i][j] = 0;
                    if_need_add_num = 1;
                }
                /*情况3*/
                else                                //k项不为空,也不等于j项,此时两个都不需要动,只是下标需要变换
                    board[i][++k] = board[i][j];    //把j项移到k项的紧挨着的右边

                if (j != k)                            //移动过之后不相等说明之前他们不是紧挨着的
                {
                    board[i][j] = 0;
                    if_need_add_num = 1;            //此时移动虽然没有消去一个数,但是也要添加一个随机数出来
                }
            }
        }
    }
}

项目完整代码在原博客中已经给出。

运行效果:

用c语言编写2048小游戏_2048小游戏c语言课程设计

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/166717.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 淘宝中间件团队博客怎么做_应用服务器中间件

    淘宝中间件团队博客怎么做_应用服务器中间件http://jm.taobao.org-淘宝中间件团队博客

    2022年8月13日
    3
  • 使用CSS画一个三角形

    使用CSS画一个三角形使用CSS画一个三角形

    2022年6月15日
    27
  • Matlab中的plotyy用法总结[通俗易懂]

    Matlab中的plotyy用法总结[通俗易懂]在梳理数据,或者写论文的时候往往需要画出两个纵坐标不同的图,以显现两组数据的不同变化,在这种情况下,一般两组数据的横坐标均是相同的,但是纵坐标确实不是一个数量级的,所以为了更好的表现两组数据的变化趋势,需要用到plotyy函数绘制双纵坐标的二维图,具体用法总结如下:调用格式:1.plotyy(X1,Y1,X2,Y2):以左、右不同纵轴绘制X1-Y1、X2-Y2两条曲线。2.plotyy(…

    2022年6月29日
    55
  • python光流法算法学习「建议收藏」

    python光流法算法学习「建议收藏」基于python-opencv程序对光流法的理解光流法的定义Lucas-Kanade光流原理Shi-Tomasi角点检测python-opencv代码demo光流法的定义光流法是空间运动物体在观察成像平面上的像素运动的瞬时速度,是利用图像序列中像素在时间域上的变化以及相邻帧之间的相关性来找到上一帧跟当前帧之间存在的对应关系,从而计算出相邻帧之间物体的运动信息的一种方法。一般而言,光流是由于场景…

    2022年7月23日
    6
  • gmap绘制路径_cad画可见点

    gmap绘制路径_cad画可见点将在地图上用鼠标左键点击的点绘制出来(两点之间用直线连接),并且能够随地图一起实现等比例缩放和拖拽。GMap主要涉及到几个坐标的转换:经纬度与GPoint的转换:GPoint是gMapControl控件坐标,坐标原点(0,0)位于控件的左上角,这个转换用函数FromLatLngToLocal()完成。GPoint与经纬度的转换:将控件坐标转换成经纬度,用函数FromLocalToL

    2025年7月30日
    0
  • Android手机的像素密度(dpi)计算

    Android手机的像素密度(dpi)计算(1)分辨率。分辨率就是手机屏幕的像素点数,一般描述成屏幕的“宽×高”,安卓手机屏幕常见的分辨率有480×800、720×1280、1080×1920等。720×1280表示此屏幕在宽度方向有720个像素,在高度方向有1280个像素。(2)屏幕大小。屏幕大小是手机对角线的物理尺寸,以英寸(inch)为单位。比如某某手机为“5寸大屏手机”,就是指对角线的尺寸,5寸×2.54厘米/寸=12.7厘米。…

    2022年5月29日
    45

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号