//-------------------------------------------【头文件及引用】----------------------------------------------------//
#include <Windows.h>
#include <time.h>
#include <stdlib.h>
#include <vector>
#include <map>
#include <string>
#include <iostream>
#pragma comment(lib,"winmm.lib")
#pragma comment(lib,"Msimg32.lib")
using namespace std; //-------------------------------------------【宏定义】----------------------------------------------------------//
#define WINDOW_WIDTH 800
#define WINDOW_HEIGHT 600
#define WINDOW_TITLE L"窗口" //-------------------------------------------【全局变量声明部分】------------------------------------------------//
HDC g_hdc = NULL, g_mdc = NULL, g_bufdc = NULL; //全局设备环境句柄
HBITMAP g_hBitMap = NULL, g_hNumber[7] = {NULL}; //位图句柄
HFONT g_hFont;//文字句柄 int Map[4][4];
int score;
const int WinFlag = 2048;
int IsOver; struct node
{
int ChangeMap[4][4];//操作后的数组
int GetScore;//得分
int flag;//操作是否成功
node(int CM[][4], int scr, int f)
{
memcpy(ChangeMap, CM, sizeof ChangeMap);
GetScore = scr;
flag = f;
}
node() {}
}; //-------------------------------------------【全局函数声明部分】------------------------------------------------//
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam);
BOOL Game_Init(HWND hwnd);
VOID Game_Paint(HWND hwnd);
BOOL Game_CleanUp(HWND hwnd); VOID Map_Init();
//四种操作(主要对数组进行旋转)
node up();
node down();
node left();
node right();
//统一转成左移后操作
node change(int a[4][4]); void InsertNumber();//随机插入数字
int judge();//判断矩阵情况,0为无法继续移动,1为可以继续移动但未达到胜利标准,2为达到胜利标准 //-------------------------------------------【 main函数 】------------------------------------------------//
int main() {
HINSTANCE hInstance = GetModuleHandle(NULL);
int nShowCmd = true;
WNDCLASSEX wndClass = {
sizeof(WNDCLASSEX),
CS_HREDRAW | CS_VREDRAW,
WndProc,
0L,
0L,
hInstance,
(HICON)::LoadImage(NULL,L"icon.ico", IMAGE_ICON, 0, 0, LR_DEFAULTSIZE | LR_LOADFROMFILE),
LoadCursor(NULL, IDC_ARROW),
(HBRUSH)GetStockObject(GRAY_BRUSH),
NULL,
L"ForTheDream",//窗口类的名称
NULL
};
if( !RegisterClassEx(&wndClass)) return -1;
HWND hWnd = CreateWindow( L"ForTheDream", WINDOW_TITLE, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT, WINDOW_WIDTH, WINDOW_HEIGHT, NULL, NULL, hInstance, NULL);
MoveWindow(hWnd, 250, 80, WINDOW_WIDTH, WINDOW_HEIGHT, true);
ShowWindow(hWnd, nShowCmd);
UpdateWindow(hWnd); if(!Game_Init(hWnd)) return -1; //消息循环过程
MSG msg = {0};
while(msg.message != WM_QUIT) {
if(PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) {
TranslateMessage(&msg); //虚拟键消息转换成字符消息
DispatchMessage(&msg); //分发一个消息给窗口程序
}
else
{
Game_Paint(hWnd);
}
}
UnregisterClass(L"ForTheDream", wndClass.hInstance );//注销 return 0;
} //---------------------------------------【窗口过程函数WndProc( )部分】------------------------------------------//
//描述:窗口过程函数,对窗口消息进行处理
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) {
//定义一个PAINTSTRUCT结构体来记录一些绘制信息
PAINTSTRUCT paintStruct;
switch(message) {
case WM_KEYDOWN:
{
if(wParam == VK_ESCAPE)
{
if(MessageBox(hwnd, L"确认退出吗?", L"退出", MB_YESNO) == IDYES)
DestroyWindow(hwnd);
}
else if((wParam == VK_UP || wParam == VK_DOWN ||wParam == VK_LEFT ||wParam == VK_RIGHT) && !IsOver)
{
node res;
switch(wParam)
{
case VK_UP:
res = up();
break;
case VK_DOWN:
res = down();
break;
case VK_LEFT:
res = left();
break;
case VK_RIGHT:
res = right();
break;
}
memcpy(Map, res.ChangeMap, sizeof Map);
score += res.GetScore;
if(res.flag)
InsertNumber();
int JudgeRes = judge();
if(JudgeRes == 2)
{
IsOver = true;
Game_Paint(hwnd);
if(MessageBox(hwnd, L"恭喜你胜利了!是否重新开始游戏?", L"恭喜!", MB_YESNO) == IDYES)
Map_Init();
}
else if(JudgeRes == 0)
{
IsOver = true;
Game_Paint(hwnd);
if(MessageBox(hwnd, L"哈哈哈输了吧你个菜逼!是否重新开始游戏?", L"失败", MB_YESNO) == IDYES)
Map_Init();
}
//MessageBox(hwnd,L"键盘",L"Mouse",MB_OK);
}
}
break;
case WM_DESTROY:
Game_CleanUp(hwnd);
PostQuitMessage(0);
break;
case WM_LBUTTONDOWN://鼠标消息
//MessageBox(hwnd,L"鼠标左键已按下",L"Mouse",MB_OK);
break;
default:
return DefWindowProc(hwnd, message, wParam, lParam);
}
return 0;
} //---------------------------------------【游戏初始化部分】------------------------------------------//
//描述:贴背景贴图
BOOL Game_Init(HWND hwnd)
{
HBITMAP bmp;
Map_Init();
g_hdc = GetDC(hwnd);//获取设备环境句柄k
g_hBitMap = (HBITMAP)LoadImage(NULL, L"back.bmp", IMAGE_BITMAP, 800, 600, LR_LOADFROMFILE);//加载位图
wchar_t filename[20];
for(int i = 0; i < 11; i++)
{
memset(filename, 0, sizeof(filename));
swprintf_s(filename, L"color%d.bmp", i);
g_hNumber[i] = (HBITMAP)LoadImage(NULL, filename, IMAGE_BITMAP, 80, 80, LR_LOADFROMFILE);
}
g_mdc = CreateCompatibleDC(g_hdc);//建立兼容设备环境的内存DC
g_bufdc = CreateCompatibleDC(g_hdc);
bmp = CreateCompatibleBitmap(g_hdc, WINDOW_WIDTH, WINDOW_HEIGHT);//建立一个与窗口兼容的空的位图对象
SelectObject(g_mdc, bmp);
Game_Paint(hwnd);
return TRUE;
} VOID Game_Paint(HWND hwnd)
{
SelectObject(g_bufdc, g_hBitMap);//将位图对象选入到g_mdc内存DC中
BitBlt(g_mdc, 0, 0, WINDOW_WIDTH, WINDOW_HEIGHT, g_bufdc, 0, 0, SRCCOPY);//采用BitBlt函数贴图,参数设置为窗口大小 g_hFont = CreateFont(40, 0, 0, 0, 0, 0, 0, 0, GB2312_CHARSET, 0, 0, 0, 0, L"微软雅黑");
SelectObject(g_mdc, g_hFont);
SetBkMode(g_mdc, TRANSPARENT);
wchar_t TextScore[20];
swprintf_s(TextScore, L"%d", score);
TextOut(g_mdc, 580, 225, TextScore, wcslen(TextScore));
DeleteObject(g_hFont);
for(int i = 0; i < 4; i++)
{
for(int j = 0; j < 4; j++)
{
if(Map[i][j])
{
int NO = 0;
int tmp = Map[i][j];
while(tmp != 2)
{
NO++;
tmp /= 2;
}
SelectObject(g_bufdc, g_hNumber[NO]);
BitBlt(g_mdc, 110 + j * 100, 110 + i * 100, 80, 80, g_bufdc, 0, 0, SRCCOPY);
}
}
}
BitBlt(g_hdc,0,0,WINDOW_WIDTH,WINDOW_HEIGHT,g_mdc,0,0,SRCCOPY);
} //---------------------------------------【Game_CleanUp()函数】------------------------------------------//
//描述:资源清理函数
BOOL Game_CleanUp(HWND hwnd)
{
DeleteObject(g_hBitMap);
for(int i = 0; i < 7; i++) {
DeleteObject(g_hNumber[i]);
}
DeleteDC(g_mdc);
DeleteDC(g_bufdc);
ReleaseDC(hwnd, g_hdc);//释放设备环境
return TRUE;
} //---------------------------------------【游戏算法部分】------------------------------------------//
VOID Map_Init()
{
score = 0;
memset(Map, 0, sizeof Map);
InsertNumber();
InsertNumber();
//Map[0][0] = 1024, Map[0][1] = 1024;
IsOver = false;
} void InsertNumber()
{
srand((unsigned int)time(NULL));
int x, y;
x = rand() % 4;
y = rand() % 4;
while(Map[x][y])
{
x = rand() % 4;
y = rand() % 4;
}
int number = rand() % 10;
if(number)
Map[x][y] = 2;
else
Map[x][y] = 4;
}
node change(int a[4][4])
{
vector <int> v[4];
vector <int> tmp[4];
int ans[4][4] = {0};
int GetScore = 0;
for(int i = 0; i < 4; i++)
{
for(int j = 0; j < 4; j++)
{
if(a[i][j])
v[i].push_back(a[i][j]);
}//去0
while(v[i].size() > 1)
{
if(v[i][0] == v[i][1])
{
tmp[i].push_back(v[i][0] * 2);
GetScore += v[i][0] * 2;
v[i].erase(v[i].begin(), v[i].begin() + 2);
}//比较是否可以合并,如果可以合并到新容器中并删除已合并的两个数
else
{
tmp[i].push_back(v[i][0]);
v[i].erase(v[i].begin(), v[i].begin() + 1);
}//如果不可以合并把第一个放入容器中,删掉第一个数
}
if(v[i].size())
tmp[i].push_back(v[i][0]);//如果有剩余的数也放进去
int j = 0;
for(; j < tmp[i].size(); j++)
ans[i][j] = tmp[i][j];//存进数组
for(; j < 4; j++)
ans[i][j] = 0;//补零
}
int flag = 0;
for(int i = 0; i < 4; i++)
{
for(int j = 0; j < 4; j++)
if(ans[i][j] != a[i][j])
flag = 1;//检查是否和原数组相同,如果相同说明不可以移动
}
return node(ans, GetScore, flag);
}
node up()
{
int a[4][4];
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
a[i][j] = Map[j][3 - i];
}
node res = change(a);
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
a[i][j] = res.ChangeMap[3 - j][i];
}
return node(a, res.GetScore, res.flag);
}
node down()
{
int a[4][4];
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
a[i][j] = Map[3 - j][i];
}
node res = change(a);
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
a[i][j] = res.ChangeMap[j][3 - i];
}
return node(a, res.GetScore, res.flag);
}
node left()
{
return change(Map);
}
node right()
{
int a[4][4];
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
a[i][j] = Map[i][3 - j];
}
node res = change(a);
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
a[i][j] = res.ChangeMap[i][3 - j];
return node(a, res.GetScore, res.flag);
}
int judge()
{
for(int i = 0; i < 4; i++)
for(int j = 0; j < 4; j++)
{
if(Map[i][j] == WinFlag)
return 2;
}
int flag = 0;
flag = max(flag, up().flag);
flag = max(flag, down().flag);
flag = max(flag, left().flag);
flag = max(flag, right().flag);
return flag;
}

只是一只大作业……嗯哼  

github不知道为啥安不上了……先把代码存一下……万一又异常了咋整【擦泪

gdi写的2048的更多相关文章

  1. 280行代码:Javascript 写的2048游戏

    2048 原作者就是用Js写的,一直想尝试,但久久未动手. 昨天教学生学习JS代码.最好还是就做个有趣的游戏好了.2048这么火,是一个不错的选择. 思路: 1. 数组 ,2维数组4x4 2. 移动算 ...

  2. C语言写的2048小游戏

    基于"基于C_语言的2048算法设计_颜冠鹏.pdf" 这一篇文献提供的思路 在中国知网上能找到 就不贴具体内容了 [摘 要] 针对2048的游戏规则,分析了该游戏的算法特点,对其 ...

  3. 纯JS写的2048游戏,分享之

    这几天玩儿着2048这个游戏,突然心血来潮想练习下敲代码的思路.于是乎就模仿做了一个,到眼下位置还没有实现动态移动,不是非常好看,只是玩儿着自己模仿的小游戏还是蛮爽的,哈哈 假设没有玩儿过这个游戏,最 ...

  4. 学习js与css 写个2048

    学习阶段,还是写点小东西练练手学的有意思一点,今天用栅格布局做了一个2048,但是移动动画和合并特效没有做,只简单的实现了一下功能. 记录一下学习的过程. 1.入口函数,初始化界面,我这里是直接是一个 ...

  5. 一起来写2048(160行python代码)

    前言: Life is short ,you need python. --Bruce Eckel 我与2048的缘,不是缘于一个玩家,而是一次,一次,重新的ACM比赛.四月份校赛初赛,第一次碰到20 ...

  6. 一起写2048(160行python代码)

    前言: Life is short ,you need python. --Bruce Eckel 我与2048的缘,不是缘于一个玩家.而是一次,一次,重新的ACM比赛.四月份校赛初赛,第一次碰到20 ...

  7. 使用 .NET WinForm 开发所见即所得的 IDE 开发环境,实现不写代码直接生成应用程序

    直接切入正题,这是我09年到11年左右业余时间编写的项目,最初的想法很简单,做一个能拖拖拽拽就直接生成应用程序的工具,不用写代码,把能想到的业务操作全部封装起来,通过配置的方式把这些业务操作组织起来运 ...

  8. 用javascript实现一个2048游戏

    早就想自己写一个2048游戏了,昨晚闲着没事,终于写了一个 如下图,按方向键开始玩吧. 如果觉得操作不方便,请直接打开链接玩吧: http://gujianbo.1kapp.com/2048/2048 ...

  9. JavaScript写一个拼图游戏

    拼图游戏的代码400行, 有点多了, 在线DEMO的地址是:打开: 因为使用canvas,所以某些浏览器是不支持的: you know: 为什么要用canvas(⊙o⊙)?  因为图片是一整张jpg或 ...

随机推荐

  1. WCF 动态生成 不用增加引用两种方式

    一.fromwork2.0低版本方式 1,打开vs的命令工具 输入:wsdl wcf地址 + /l:cs /out:文件名 上面红色部分替换掉就行,文件名,你想叫什么文件名都行. 2,回车,生成的文件 ...

  2. Contest2037 - CSU Monthly 2013 Oct (problem F :ZZY and his little friends)

    http://acm.csu.edu.cn/OnlineJudge/problem.php?cid=2037&pid=5 [题解]: 没想通这题暴力可以过.... [code]: #inclu ...

  3. nginx+ tomcat集群+动静资源分离

    不知道为什么这个随便删不掉,写了也值显示一半一半不显示, 我把重新写了一遍: nginx + tomcat集群和动静资源分离

  4. ExtJS4.2学习(九)属性表格控件PropertyGrid(转)

    鸣谢网址:http://www.shuyangyang.com.cn/jishuliangongfang/qianduanjishu/2013-11-15/178.html ------------- ...

  5. js中批量处理样式——cssText的使用

    http://www.cnblogs.com/snandy/archive/2011/03/12/1980444.html

  6. [LeetCode]Divide Two Integer

    Divide two integers without using multiplication, division and mod operator. 思考:位运算.AC的时候真的想说一句“尼玛.. ...

  7. MVC 文件上传和图片剪辑

    http://www.cnblogs.com/hinton/archive/2012/03/01/2375465.html http://gallery.kissyui.com/uploader/1. ...

  8. 【转】linux下cvs配置

    1.  验证是否已安装CVS #rpm -q cvs 如果能显示出类似这样的版本信息,证明已安装CVS: #cvs-1.11.19 若没有安装信息,则需要从htttp://www.cvshome.or ...

  9. 阿里巴巴Java面试题

    研二是需要找实习的时候了,因阿里有同学内推就直接参加了电话面试,不说其他的废话直接上问题,阿里的面试官还是不错的,和蔼可亲,为人谦虚,大牛什么都懂.(投的职位是java研发)1.java中所有类的父类 ...

  10. 哈希值识别工具hash-identifier

    Hash Identifier可以用来识别各种类型的哈希值.在kali上使用方法很简单 (1)搜索hash-identifier (2)在HASH后面输入要识别的hash内容 (3)识别成功 wind ...