【小游戏】C++打造2048控制台版本

2024-04-11 02:58

本文主要是介绍【小游戏】C++打造2048控制台版本,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

翻了一下以前的代码,发现去年国庆写的2048有点意思,于是搬过来纪念一下

#include<iostream>
#include<cstdio>
#include<time.h>
#include<cstdlib>
#include <conio.h>//kbhit()函数
//宏定义----------------------------------------------------------
#define JUG_empty 1//juge结果
#define JUG_merge 2//juge结果
#define JUG_collision 3//juge结果
#define JUG_nonum 4//juge结果#define MOV_empty 1//move类型匹配
#define MOV_merge 2//move类型匹配
#define MOV_noaction 3//move类型匹配
#define MOV_auto 4//move类型匹配//命名空间声明----------------------------------------------------
using namespace std;
//自定义数据类型--------------------------------------------------
enum n_dir {Left,Right,Up,Down};
//方向类型//函数声明--------------------------------------------------------
int Move(n_dir dir,int move=1);//返回MOV_类型的结果。move表示是否移动
//块移动
int Random(int min,int max);
//取随机整数
void StartScr();
//开始界面
void GameScr();
//游戏界面
void prinum(int);
//按照固定大小打印数字
int Juge(short &num,short &des,short move = MOV_auto);
//块移动
void Welcome();
//欢迎界面
void Padding(int);
//空格填充
//全局变量--------------------------------------------------------
short wide=4,hight=4,//区域大小考虑到可选择性,所以用了变量,默认为4×4区域*map;//区域
unsigned grade=0;//
int main()
{srand((unsigned)time(NULL));//置随机数种子//---------------------欢迎界面---------------------Welcome();//---------------------开始界面---------------------StartScr();return 0;
}
void prinum(int a)
{//这里只考虑最大数字为四位数的所以块大小是1×4的长方形int len=0,leftblank,b=a;while(b){len++;b/=10;}leftblank=(4-len)/2;int i;for(i=1;i<=leftblank;i++)putchar(' ');cout<<a;for(i=1;i<=4-len-leftblank;i++)putchar(' ');
}
int Random(int min,int max)
{return rand()%(max-min+1)+min;
}
void StartScr()
{//---------------------开始游戏---------------------//以下将一维数组二维化:[i][j]=[i*wide+j]map = new short [wide*hight];//设置区域大小grade=0;//分数归零for(int i=0;i<hight;i++){for(int j=0;j<wide;j++){map[i*wide+j]=-1;//无数的标志}}map[Random(0,wide*hight-1)]=2;//map记录数值GameScr();delete []map;
}
void GameScr()
{int key1,key2,result;system("cls");//清屏int i=0;cout<<"---------";prinum(grade);cout<<"----------"<<endl;for(i=0;i<=wide;i++)cout<<"----";cout<<"----"<<endl;for(i=0;i<wide*hight;i++){if(i%wide==0)cout<<"|   ";if(map[i]==-1)cout<<"    ";elseprinum(map[i]);if((i+1)%wide==0)cout<<"   |"<<endl;//行尾}for(i=0;i<=wide;i++)cout<<"-----";while(1){if(kbhit())//当键盘被按下{key1=getch();if(key1==224)//方向键{key2=getch();//键值if(key2==72)//上result=Move(Up);else if(key2==80)//下result=Move(Down);else if(key2==75)//左result=Move(Left);else if(key2==77)//右result=Move(Right);}else //数字键{if(key1=='w'||key1=='W')//上result=Move(Up);else if(key1=='s'||key1=='S')//下result=Move(Down);else if(key1=='a'||key1=='A')//左result=Move(Left);else if(key1=='d'||key1=='D')//右result=Move(Right);}//---------------------刷新显示---------------------system("cls");//清屏int i=0;cout<<"---------";prinum(grade);cout<<"----------"<<endl;for(i=0;i<=wide;i++)cout<<"----";cout<<"----"<<endl;for(i=0;i<wide*hight;i++){if(i%wide==0)cout<<"|   ";if(map[i]==-1)cout<<"    ";elseprinum(map[i]);if((i+1)%wide==0)cout<<"   |"<<endl;//行尾}for(i=0;i<=wide;i++)cout<<"-----";if(Move(Up,0)==MOV_noaction && Move(Down,0)==MOV_noaction && Move(Left,0)==MOV_noaction && Move(Right,0)==MOV_noaction){//已经不能移动了cout<<endl<<"----------GameOver----------"<<endl;}}}
}
int Move(n_dir dir,int move)
{int i,j,k,num,des,result,count,moveresult=MOV_noaction;short *list;if(dir == Up){list = new short [hight];//记录合并位置for(j=0;j<wide;j++){count=0;for(i=1;i<hight;i++)//行上移{if(map[i*wide+j]!=-1){num=i;des=i-1;while(1)//跳过空位置{result=Juge(map[num*wide+j],map[des*wide+j],MOV_noaction);//返回判断结果if(result == JUG_empty)//判断为空位置{moveresult=MOV_empty;if(move)Juge(map[num*wide+j],map[des*wide+j]);//移到空位置des--;num--;if(des<0)//目标超界跳出break;continue;}else if(result == JUG_merge)//结果为合并{for(k=0;k<count;k++)if(des==list[k])//目标位置已经被合并过了goto up_out;moveresult=MOV_merge;if(move){grade+=map[num*wide+j];Juge(map[num*wide+j],map[des*wide+j]);//合并list[count++]=des;//记录合并位置}up_out:break;}elsebreak;}}}}delete []list;}else if(dir == Down){list = new short [hight];//记录合并位置for(j=0;j<wide;j++){count=0;for(i=hight-2;i>=0;i--)//行下移{if(map[i*wide+j]!=-1){num=i;des=i+1;while(1)//跳过空位置{result=Juge(map[num*wide+j],map[des*wide+j],MOV_noaction);//返回判断结果if(result == JUG_empty)//判断为空位置{moveresult=MOV_empty;if(move)Juge(map[num*wide+j],map[des*wide+j]);//移到空位置des++;num++;if(des>hight-1)//目标超界跳出break;continue;}else if(result == JUG_merge)//结果为合并{for(k=0;k<count;k++)if(des==list[k])//目标位置已经被合并过了goto down_out;moveresult=MOV_merge;if(move){grade+=map[num*wide+j];Juge(map[num*wide+j],map[des*wide+j]);//合并list[count++]=des;//记录合并位置}down_out:break;}elsebreak;}}}}delete []list;}else if(dir == Left){list = new short [wide];//记录合并位置for(i=0;i<hight;i++){count=0;for(j=1;j<wide;j++)//行左移{if(map[i*wide+j]!=-1){num=j;des=j-1;while(1)//跳过空位置{result=Juge(map[i*wide+num],map[i*wide+des],MOV_noaction);//返回判断结果if(result == JUG_empty)//判断为空位置{moveresult=MOV_empty;if(move)Juge(map[i*wide+num],map[i*wide+des]);//移到空位置des--;num--;if(des<0)//目标超界跳出break;continue;}else if(result == JUG_merge)//结果为合并{for(k=0;k<count;k++)if(des==list[k])//目标位置已经被合并过了goto left_out;moveresult=MOV_merge;if(move){grade+=map[i*wide+num];Juge(map[i*wide+num],map[i*wide+des]);//合并list[count++]=des;//记录合并位置}left_out:break;}elsebreak;}}}}delete []list;}else if(dir == Right){list = new short [wide];//记录合并位置for(i=0;i<hight;i++){count=0;for(j=wide-2;j>=0;j--)//行左移{if(map[i*wide+j]!=-1){num=j;des=j+1;while(1)//跳过空位置{result=Juge(map[i*wide+num],map[i*wide+des],MOV_noaction);//返回判断结果if(result == JUG_empty)//判断为空位置{moveresult=MOV_empty;if(move)Juge(map[i*wide+num],map[i*wide+des]);//移到空位置des++;num++;if(des<0)//目标超界跳出break;continue;}else if(result == JUG_merge)//结果为合并{for(k=0;k<count;k++)if(des==list[k])//目标位置已经被合并过了goto right_out;moveresult=MOV_merge;if(move){grade+=map[i*wide+num];//加分Juge(map[i*wide+num],map[i*wide+des]);//合并list[count++]=des;//记录合并位置}right_out:break;}elsebreak;}}}}delete []list;}if(move&&moveresult!=MOV_noaction){int posi=Random(0,wide*hight-1);while(map[posi]!=-1)//保证是空位posi=Random(0,wide*hight-1);if(Random(1,8)<8)//2的几率:4的几率=8:1map[posi]=2;elsemap[posi]=4;}return moveresult;
}int Juge(short &num,short &des,short move)//num一定不是-1
{if(des==-1)//目的地无数{if(move==MOV_auto||move==MOV_empty){des=num;//数值转移num=-1;//置空}return JUG_empty;}else //目的地有数{if(num==des)//数值一样{if(move==MOV_auto||move==MOV_merge){des+=num;//合并num=-1;//置空}return JUG_merge;}elsereturn JUG_collision;}}
//
//欢迎界面
//
void Welcome()
{Padding(-10);printf("欢迎来到2048!!!\n");Padding(-10);printf("上下左右为wsad键或方向键,感受这款经典游戏带来的快感吧!\n");system("pause");
}
//
//显示一定数量的空格
//参数1:相对偏移
void Padding(int shift)
{int total=20,i;total+=shift;for(i=0;i<total;i++){putchar(' ');}
}


这篇关于【小游戏】C++打造2048控制台版本的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/892899

相关文章

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

python版本切换工具pyenv的安装及用法

《python版本切换工具pyenv的安装及用法》Pyenv是管理Python版本的最佳工具之一,特别适合开发者和需要切换多个Python版本的用户,:本文主要介绍python版本切换工具pyen... 目录Pyenv 是什么?安装 Pyenv(MACOS)使用 Homebrew:配置 shell(zsh

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias

Python使用wxPython打造文件批量压缩工具

《Python使用wxPython打造文件批量压缩工具》在日常工作中,我们经常需要将多个文件打包压缩,下面我们就来看看如何使用Python的wxPython库,创建一个功能强大的图形化文件批量压缩工具... 目录前言项目需求分析技术选型运行效果核心实现1. 界面布局设计2. 多种文件添加方式3. 可勾选的文