Android 之 2048 的游戏逻辑分析
继续学习了极客学院的实战路径课程,讲到了2048游戏的编写过程,我在这里作个总结分享给大家(结果会附源代码和我改写后的代码):
这里主要包括两个方面:1.2048界面的绘制 2.2048算法逻辑的实现 3.添加随机数 4.判断游戏结束
先看效果图(真机上模拟图):

1.界面的绘制
界面的绘制相对还是比较简单的。先新建一个card卡片类,这个类主要是描述在效果图中那16个小方块
/*
* 这个类主要用来初始化2048游戏中的方块
*/
public class Card extends FrameLayout{ private TextView lable;
public Card(Context context) {
super(context);
lable = new TextView(getContext());
lable.setTextSize(32);
lable.setGravity(Gravity.CENTER);
lable.setBackgroundColor(0X30FFFFFF); //30表示透明度,透明度范围是00-ff,后六位是颜色值 //下面设置了Layout_wight和Layout_height分别为match_parent(-1代表match_parent,-2代表wrap_content)
LayoutParams lp = new LayoutParams(-1,-1);
lp.setMargins(10, 10, 10, 10); //设置card的间距
addView(lable,lp);
setNum(0);
} int num = 0;
public int getNum(){
return num;
}
public void setNum(int num) {
this.num = num;
//当cardMap[][]<=0时,设为""
if (num<=0) {
lable.setText("");
}else {
lable.setText(num+"");
}
}
public boolean equals(Card o) {
return getNum()==o.getNum();
}
}
然后新建一个GameView的主类,使它继承GirdLayout,复写其中的方法。再在此之前需要更改一下main_activity布局文件,包括计分的一个textView,和在外面实现的布局。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.example.game1024.MainActivity" > <LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"> <TextView
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_gravity="center_vertical"
android:textSize="20sp"
android:text="score:"/>
<TextView
android:id="@+id/cvScore"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="20sp"
/>
</LinearLayout> <com.example.game1024.GameView
android:background="#00ff00"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1">
</com.example.game1024.GameView> </LinearLayout>
public class GameView extends GridLayout{
public GameView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
initGame();
}
public GameView(Context context, AttributeSet attrs) {
super(context, attrs);
initGame();
}
public GameView(Context context) {
super(context);
initGame();
}
public void initGame(){
setColumnCount(4); //设定规定行数为四行
setBackgroundColor(0XffF4A460);
setOnTouchListener(new View.OnTouchListener() {
private float startX,startY,offsetX,offsetY;
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
//取得点击时的初始坐标
System.out.println("00000");
startX = event.getX();
startY = event.getY();
break;
case MotionEvent.ACTION_UP:
//求得各方向的偏移量
System.out.println("0--0");
offsetX = event.getX() - startX;
offsetY = event.getY() - startY;
/*
* 比较在x轴和y轴的偏移量,可以判断为上下滑动,还是左右滑动
* 接着判断偏移量的正负,来判断是具体向哪个方向的滑动
* 下面我写移动的顺序为:向左,右,上,下
*/
if (Math.abs(offsetX)>Math.abs(offsetY)) {
//当移动的距离大于5dp时才看作移动
if (offsetX<-5) {
swipeLeft();
}else if (offsetX>5) {
swipeRight();
}
}else if (Math.abs(offsetX)<Math.abs(offsetY)) {
if (offsetY<-5) {
swipeUp();
}else if (offsetY>5) {
swipeDown();
}
}
break;
default:
break;
}
return true;
}
});
}
/*
* (non-Javadoc)
* @see android.view.View#onSizeChanged(int, int, int, int)
* 当屏幕的宽高改变时,卡片所占的宽高会随之改变
*/
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
int cardWidth =(Math.min(w, h)-10)/4;
addCard(cardWidth, cardWidth);
startGame();
}
//建立一个数组用于存储各个卡片的num
private Card cardMap[][] = new Card[4][4];
/*
* 添加card卡片的方法
*/
private void addCard(int cardwidth, int cardheight){
Card c;
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) {
c = new Card(getContext());
c.setNum(0);
addView(c,cardwidth,cardheight);
cardMap[x][y]=c;
}
}
}
2.游戏逻辑的算法
这个注释比较清楚,大家看注释吧 ,if(judge)是后面要判断游戏结束时用的
/*
* 移动方面的算法
*/
private void swipeLeft(){
boolean judge = false;
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) { for (int x1 = x+1; x1 < 4; x1++) {
//先判断在当前点的右边是否存在num大于零的点,如果是,进行下一步,否的话继续循环
//判断当前点是否为空(小于等于零相当于空,在card类中已经设置为了不显示),为空:将右面的点赋予当前点,同时自身置零。不为空的话判断当前点是否和
//右面的点相等,相等将当前点的num乘2,并将右面的点置零
if (cardMap[x1][y].getNum()>0) { if (cardMap[x][y].getNum()<=0) {
cardMap[x][y].setNum(cardMap[x1][y].getNum());
cardMap[x1][y].setNum(0);
/*
* 假如在第1个位置有一个数,它将移动到第0个位置,此时继续循环,如果第2个位置也有数,
* 此时因为第0个位置已经存在了数值,所以第2个位置的数不会移动到第一个空白位置,而是保持不变
* 因此需要将x减1,在比较一遍
*/
x--;
judge = true;
}else if (cardMap[x][y].getNum()==cardMap[x1][y].getNum()) {
cardMap[x][y].setNum(cardMap[x][y].getNum()*2);
cardMap[x1][y].setNum(0); MainActivity.getMainActivity().addScore(cardMap[x][y].getNum());
judge = true;
}
break;
}
}
}
}
if (judge) {
checkGame();
addRandom();
}
}
private void swipeRight(){
boolean judge = false;
for (int y = 0; y < 4; y++) {
for (int x = 3; x >= 0; x--) { for (int x1 = x-1; x1 >=0; x1--) {
if (cardMap[x1][y].getNum()>0) { if (cardMap[x][y].getNum()<=0) {
cardMap[x][y].setNum(cardMap[x1][y].getNum());
cardMap[x1][y].setNum(0);
x++;
judge = true;
}else if (cardMap[x][y].getNum()==cardMap[x1][y].getNum()) {
cardMap[x][y].setNum(cardMap[x][y].getNum()*2);
cardMap[x1][y].setNum(0); MainActivity.getMainActivity().addScore(cardMap[x][y].getNum());
judge = true;
}
break;
}
}
}
}
if (judge) {
checkGame();
addRandom();
}
}
private void swipeUp(){
boolean judge = false;
for (int x = 0; x < 4; x++) {
for (int y = 0; y <4; y++) { for (int y1 = y+1 ; y1 <4 ; y1++) {
if (cardMap[x][y1].getNum()>0) { if (cardMap[x][y].getNum()<=0) {
cardMap[x][y].setNum(cardMap[x][y1].getNum());
cardMap[x][y1].setNum(0);
y--;
judge = true;
}else if (cardMap[x][y].getNum()==cardMap[x][y1].getNum()) {
cardMap[x][y].setNum(cardMap[x][y].getNum()*2);
cardMap[x][y1].setNum(0); MainActivity.getMainActivity().addScore(cardMap[x][y].getNum());
judge = true;
}
break;
}
}
}
}
if (judge) {
checkGame();
addRandom();
}
}
private void swipeDown(){
boolean judge = false;
for (int x = 0; x < 4; x++) {
for (int y = 3; y >=0; y--) { for (int y1 = y-1 ; y1 >=0; y1--) {
if (cardMap[x][y1].getNum()>0) { if (cardMap[x][y].getNum()<=0) {
cardMap[x][y].setNum(cardMap[x][y1].getNum());
cardMap[x][y1].setNum(0);
y++;
judge = true;
}else if (cardMap[x][y].getNum()==cardMap[x][y1].getNum()) {
cardMap[x][y].setNum(cardMap[x][y].getNum()*2);
cardMap[x][y1].setNum(0); MainActivity.getMainActivity().addScore(cardMap[x][y].getNum());
judge = true;
}
break;
}
}
}
}
if (judge) {
checkGame();
addRandom();
}
}
3.添加随机出现的点:
同样在GameView中添加方法,startGame()在onSizeChanged()方法中调用
/*
* 游戏开始的初始化方法
*/
private void startGame(){
MainActivity.getMainActivity().clearScore();
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) {
cardMap[x][y].setNum(0);
}
} addRandom();
addRandom();
} private List<Point> emptyPoint =new ArrayList<Point>();
//添加随机数的方法
private void addRandom(){
emptyPoint.clear();
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) {
//将cardMap中小于0的点放在emptyPoint中
if (cardMap[x][y].getNum()<=0) {
emptyPoint.add(new Point(x, y));
}
}
}
//随机移除emptyPoint中的一个单元
Point p = emptyPoint.remove((int)(Math.random()*emptyPoint.size()));
cardMap[p.x][p.y].setNum(Math.random()>0.1?2:4);
}
4.判断游戏结束
/*
* 结束游戏方法
* 当存在点和旁边的数相等时,游戏就不会提示已经结束
*/
private void checkGame(){
boolean check = true;
ALL:
for (int y = 0; y < 4; y++) {
for (int x = 0; x < 4; x++) {
if (cardMap[x][y].getNum()==0||
x>0&&cardMap[x][y].equals(cardMap[x-1][y])||
x<3&&cardMap[x][y].equals(cardMap[x+1][y])||
y>0&&cardMap[x][y].equals(cardMap[x][y-1])||
y<3&&cardMap[x][y].equals(cardMap[x][y+1])) {
check = false;
break ALL;
}
}
} // 提示游戏失败
if (check) {
new AlertDialog.Builder(getContext()).setTitle("提示").setMessage("游戏失败").setPositiveButton("再来一次", new DialogInterface.OnClickListener() { @Override
public void onClick(DialogInterface dialog, int which) {
startGame();
}
}).show();
}
}
我改进的2048:
1.增加了最高分显示
2.增加了重新开始游戏按钮
3.增加了颜色变化
Game1024原版:http://pan.baidu.com/s/1sjpCdIl
Game2048改进版地址:http://pan.baidu.com/s/1o6qyfJc
Android 之 2048 的游戏逻辑分析的更多相关文章
- 从零開始开发Android版2048 (二)获取手势信息
今天是尝试開始Android版2048小游戏的第二天.在今天,我主要学习了怎样获取用户在屏幕滑动的手势,以及对布局进行了一些小小的完好. 获取用户操作的手势(比方向左滑.向右滑等)主要用到了Gestu ...
- Swift实战之2048小游戏
上周在图书馆借了一本Swift语言实战入门,入个门玩一玩^_^正好这本书的后面有一个2048小游戏的实例,笔者跟着实战了一把. 差不多一周的时间,到今天,游戏的基本功能已基本实现,细节我已不打算继续完 ...
- 从零開始开发Android版2048 (一)初始化界面
自学Android一个月多了,一直在工作之余零零散散地看一些东西.感觉经常使用的东西都有些了解了,可是一開始写代码总会出各种奇葩的问题.感觉还是代码写得太少.这样继续杂乱地学习下去进度也太慢了,并且学 ...
- 从零開始开发Android版2048 (四) 分数、重置、结束
这一篇的内容主要是在上一篇的基础上,增加分数计算(包含当前分数和最高分数).游戏结束的推断以及游戏界面的重置这三个部分的功能. 一.分数的计算和保存 首先,2048这个游戏的分数包含 ...
- 从零開始开发Android版2048 (三)逻辑推断
近期工作比較忙,所以更新的慢了一点,今天的主要内容是关于Android版2048的逻辑推断,经过本篇的解说,基本上完毕了这个游戏的主体部分. 首先还是看一下,我在实现2048时用到的一些存储的数据结构 ...
- jQuery实践-网页版2048小游戏
▓▓▓▓▓▓ 大致介绍 看了一个实现网页版2048小游戏的视频,觉得能做出自己以前喜欢玩的小游戏很有意思便自己动手试了试,真正的验证了这句话-不要以为你以为的就是你以为的,看视频时觉得看懂了,会写了, ...
- Android快乐贪吃蛇游戏实战项目开发教程-01项目概述与目录
一.项目简介 贪吃蛇是一个很经典的游戏,也很适合用来学习.本教程将和大家一起做一个Android版的贪吃蛇游戏. 我已经将做好的案例上传到了应用宝,无病毒.无广告,大家可以放心下载下来把玩一下.应用宝 ...
- C# 开发2048小游戏
这应该是几个月前,闲的手痒,敲了一上午代码搞出来的,随之就把它丢弃了,当时让别人玩过,提过几条更改建议,但是时至今日,我也没有进行过优化和更改(本人只会作案,不会收场,嘎嘎),下面的建议要给代码爱好的 ...
- Android版的疯狂猜图游戏源码完整版分享
这个游戏源码是在安装教程网那么分享过来的,Android版的疯狂猜图游戏源码完整版分享,也是本人之前很早以前发的一款游戏源码的,大家如果想了解一下,可以看看吧,不说多了,上一个图先吧. > ...
随机推荐
- NopCommerce之事件通知
mark下,等下写了. NewsController 控制器NewsCommentAdd()缓存清除,使用到了事件
- 网络通信分享(一):数字签名,数字证书,https通信,数据加密
加密算法: 一:对称加密算法 在对称加密算法中,加密使用的密钥和解密使用的密钥是相同的.也就是说,加密和解密都是使用的同一个密钥.因此对称加密算法要保证安全性的话,密钥要做好保密,只能让使用的人知道, ...
- 移除了css框架,世界干净了
在之前的webapp项目里,我使用了bootstrap作为三方的css库,只调取了其中一部分源码的less使用,大部分代码仍然是自己写的. 自己的代码也是参照bootstrap的目录结构和它的一些规范 ...
- 15个带给您优秀用户体验的移动应用 UI 设计
在今天在移动 App 界面设计中,你可以看到不同创意类型的视觉效果.特别是在 Dribbble 上面,有有很多移动应用程序的 UI 概念设计,让你惊叹.如果你想获得灵感,那很有必要看看下面15个优秀用 ...
- ASP.NET MVC分页实现
ASP.NET MVC中不能使用分页控件,所以我就自己写了一个分页局部视图,配合PageInfo类,即可实现在任何页面任意位置呈现分页,由于采用的是基于POST分页方式,所以唯一的限制就是必须放在FO ...
- C#函数式程序设计之局部套用与部分应用
函数式设计的核心与函数的应用以及函数如何作为算法的基本模块有关.利用局部套用技术可以把所有函数看成是函数类的成员,这些函数只有一个形参,有了局部套用,才有部分应用.部分应用是使函数模块化成为可能的两个 ...
- Linq查询操作之投影操作
投影操作,乍一看不知道在说啥.那么什么是投影操作呢?其实就是Select操作,名字起的怪怪的.和Linq查询表达式中的select操作是一样的.它能够选择数据源中的元素,并指定元素的表现形式.投影操作 ...
- 使用mvc3实现ajax跨域
ajax跨域一般两种方式 1:cors,2:jsonp, 1:cors jsonp是get形式,承载的信息量有限,所以信息量较大时CORS是不二选择 在请求消息头添头 Access-Control ...
- VS "15" 预览 5 中 VB 15 新增的功能
VS "15" 预览 5 给 VB 带来了更新.这次的更新内容有3个: * 值元组 ValueTuple这个功能能把一组计算结果成组返回.为了使用这个功能,我们要安装 System ...
- MAC OS设置JDK小结
开始折腾mac pro,在网上找了些资料,自己也尝试了一下,特在这里做个小结. mac中JDK的位置 OSX默认的JDK /System/Library/Frameworks/JavaVM.frame ...