android 的2048小游戏完整实现:GridLayout布局(android 4.0及以上)。

曾经做过一个2048的算法题,学了几天android,认为能够实现个安卓版的。也就动手写了个。

包括的东西:

GridLayout布局

在activity中动态加入view组件

推断用户在屏幕滑动的的方向

2048算法(參考之前用C++写的,写的还算通俗易懂吧,http://blog.csdn.net/liang5630/article/details/39895087)。

不多说,先上图:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbGlhbmc1NjMw/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">

package com.example.y2048;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Button;
import android.widget.GridLayout;
import android.widget.TextView;
import android.widget.Toast; import com.example.y2048.map.Direction;
import com.example.y2048.map.Maps; public class MainActivity extends Activity {
private String tag = "GridLayoutActivity";
GridLayout gridLayout;
float startX = 0, startY = 0, endX, endY;
Maps maps = new Maps();
private TextView score;
private TextView best; @SuppressLint("NewApi")
void init() {
// 获取View对象
gridLayout = (GridLayout) findViewById(R.id.root); for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
Button bn = new Button(this);
bn.setClickable(false);
bn.setText("");
// 设置该button的字号大小
bn.setTextSize(30);
bn.setWidth(120);
bn.setHeight(120);
// 指定该组件所在的行
GridLayout.Spec rowSpec = GridLayout.spec(i + 2);
// 指定该组件所在的列
GridLayout.Spec columnSpec = GridLayout.spec(j);
String msg = "rowSpec:" + (i + 2) + " - columnSpec:" + (j);
Log.d(tag, msg);
GridLayout.LayoutParams params = new GridLayout.LayoutParams(
rowSpec, columnSpec);
// 指定该组件占满容器
// params.setGravity(Gravity.FILL);
gridLayout.addView(bn, params);
maps.addButton(i, j, bn);
}
}
score = (TextView) findViewById(R.id.score);
score.setText("0");
best = (TextView) findViewById(R.id.best);
maps.setScore(score);
maps.setBest(best);
maps.init();
} @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
} @Override
public boolean dispatchTouchEvent(MotionEvent event) { // System.out.println("触摸");
int action = event.getAction();
if (action == MotionEvent.ACTION_DOWN) {
startX = event.getX();
startY = event.getY();
} else if (action == MotionEvent.ACTION_UP) {
endX = event.getX();
endY = event.getY();
int direction = GetSlideDirection(startX, startY, endX, endY);
// System.out.println(startX+","+startY+"|"+endX+","+endY+" "+direction);
// Toast.makeText(this, direction+"", Toast.LENGTH_LONG).show();
boolean gameOver = maps.Slide(direction);
if (gameOver) {
if(maps.getScore()>maps.getBestScore()){
Toast.makeText(this, "恭喜超过最佳记录!。!", Toast.LENGTH_SHORT).show();
maps.setBestScore(maps.getScore());
best.setText(maps.getScore()+"");
}else{
Toast.makeText(this, "GameOver", Toast.LENGTH_SHORT).show();
} }
}
return super.dispatchTouchEvent(event);
} // 返回角度
private double GetSlideAngle(float dx, float dy) {
return Math.atan2(dy, dx) * 180 / Math.PI;
} // 依据起点和终点返回方向 1:向上。2:向下,3:向左,4:向右,0:未滑动
private int GetSlideDirection(float startX, float startY, float endX,
float endY) {
float dy = startY - endY;
float dx = endX - startX;
int result = Direction.NONE;
// 假设滑动距离太短
if (Math.abs(dx) < 2 && Math.abs(dy) < 2) {
return result;
}
double angle = GetSlideAngle(dx, dy);
if (angle >= -45 && angle < 45) {
return Direction.RIGHT;
} else if (angle >= 45 && angle < 135) {
return Direction.UP;
} else if (angle >= -135 && angle < -45) {
return Direction.DOWN;
} else if ((angle >= 135 && angle <= 180)
|| (angle >= -180 && angle < -135)) {
return Direction.LEFT;
}
return result;
} public void reset(View view) {
maps.init();
} }
package com.example.y2048.map;

public interface Direction {
//1:向上,2:向下,3:向左,4:向右,0:未滑动
static final int LEFT=3;
static final int RIGHT=4;
static final int UP=1;
static final int DOWN=2;
static final int NONE=0;
}
package com.example.y2048.map;

import java.util.Random;

import android.content.Context;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.view.View;
import android.widget.Button;
import android.widget.TextView; public class Maps {
private TextView score;
private TextView best;
private Button[][] maps = new Button[4][4]; public void addButton(int i, int j, Button btn) {
maps[i][j] = btn;
} private void swapText(Button btn1, Button btn2) {
CharSequence text = btn1.getText();
btn1.setText(btn2.getText());
btn2.setText(text);
} void up_remove_blank() {
int i, j, k;
for (j = 0; j < 4; j++) {
for (i = 1; i < 4; i++) {
k = i;
while (k - 1 >= 0
&& maps[k - 1][j].getText().toString().length() == 0) {// 上面的那个为空
swapText(maps[k][j], maps[k - 1][j]);
k--; }
}
}
} void down_remove_blank() {
int i, j, k;
for (j = 0; j < 4; j++) {
for (i = 2; i >= 0; i--) {
k = i;
while (k + 1 <= 3
&& maps[k + 1][j].getText().toString().length() == 0) {// 上面的那个为空
swapText(maps[k][j], maps[k + 1][j]);
k++;
}
}
}
} void left_remove_blank() {
int i, j, k;
for (i = 0; i < 4; i++) {
for (j = 1; j < 4; j++) {
k = j;
while (k - 1 >= 0
&& maps[i][k - 1].getText().toString().length() == 0) {// 上面的那个为空
swapText(maps[i][k], maps[i][k - 1]);
k--;
}
}
}
} void right_remove_blank() {
int i, j, k;
for (i = 0; i < 4; i++) {
for (j = 2; j >= 0; j--) {
k = j;
while (k + 1 <= 3
&& maps[i][k + 1].getText().toString().length() == 0) {// 上面的那个为空
swapText(maps[i][k], maps[i][k + 1]);
k++;
}
}
}
} void left() {
int i, j;
for (i = 0; i < 4; i++) {
for (j = 0; j < 3; j++) {
String s1 = maps[i][j].getText().toString();
String s2 = maps[i][j + 1].getText().toString(); if (s1.equals(s2) && !s1.equals("")) {
// maps[i][j]+=maps[i][j+1];
Integer sum = Integer.valueOf(maps[i][j].getText()
.toString());
sum += Integer.valueOf(maps[i][j + 1].getText().toString()); int total = Integer.valueOf(score.getText().toString());
score.setText(String.valueOf(sum + total));
maps[i][j].setText(sum.toString());
maps[i][j + 1].setText("");
left_remove_blank();
}
}
}
} void right() {
int i, j;
for (i = 0; i < 4; i++) {
for (j = 3; j >= 1; j--) {
String s1 = maps[i][j].getText().toString();
String s2 = maps[i][j - 1].getText().toString();
if (s1.equals(s2) && !s1.equals("")) {
// maps[i][j]+=maps[i][j-1];
// maps[i][j-1]=0;
Integer sum = Integer.valueOf(maps[i][j].getText()
.toString());
sum += Integer.valueOf(maps[i][j - 1].getText().toString());
int total = Integer.valueOf(score.getText().toString());
score.setText(String.valueOf(sum + total));
maps[i][j].setText(sum.toString());
maps[i][j - 1].setText("");
right_remove_blank();
}
}
}
} void up() {
int i, j;
for (j = 0; j < 4; j++) {// 每一列
for (i = 0; i < 3; i++) {
String s1 = maps[i][j].getText().toString();
String s2 = maps[i + 1][j].getText().toString();
if (s1.equals(s2) && !s1.equals("")) {
// maps[i][j]=maps[i][j]+maps[i+1][j];
// maps[i+1][j]=0;
Integer sum = Integer.valueOf(maps[i][j].getText()
.toString());
sum += Integer.valueOf(maps[i + 1][j].getText().toString());
int total = Integer.valueOf(score.getText().toString());
score.setText(String.valueOf(sum + total));
maps[i][j].setText(sum.toString());
maps[i + 1][j].setText("");
// 移除空格
up_remove_blank();
}
}
}
} void down() {
int i, j;
for (j = 0; j < 4; j++) {// 每一列
for (i = 3; i >= 1; i--) {
String s1 = maps[i][j].getText().toString();
String s2 = maps[i - 1][j].getText().toString();
if (s1.equals(s2) && !s1.equals("")) {
// maps[i][j]=maps[i][j]+maps[i-1][j];
// maps[i-1][j]=0;
Integer sum = Integer.valueOf(maps[i][j].getText()
.toString());
sum += Integer.valueOf(maps[i - 1][j].getText().toString());
int total = Integer.valueOf(score.getText().toString());
score.setText(String.valueOf(sum + total));
maps[i][j].setText(sum.toString());
maps[i - 1][j].setText("");
// 移除空格
down_remove_blank();
}
}
}
} private void addNumber() {
Random random = new Random();
int x = random.nextInt(4);
int y = random.nextInt(4);
int number = random.nextInt(20);//出现2的概率为95% 4的概率5%
if(number==0) number=4;
else number=2;
while (maps[x][y].getText().toString().length() != 0) {
x = random.nextInt(4);
y = random.nextInt(4);
}
maps[x][y].setText(number + "");
} public void init() {
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
maps[i][j].setText("");
}
}
score.setText("0");
addNumber();
addNumber();
} private boolean isFull() {
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
if (maps[i][j].getText().toString().length() == 0) {
return false;
}
}
}
return true;
} public boolean Slide(int direction) {
if (direction == Direction.LEFT) {
left_remove_blank();
left();
if (isFull())
return true;
else {
addNumber();
}
} else if (direction == Direction.RIGHT) {
right_remove_blank();
right();
if (isFull())
return true;
else {
addNumber();
}
} else if (direction == Direction.UP) {
up_remove_blank();
up();
if (isFull())
return true;
else {
addNumber();
}
} else if (direction == Direction.DOWN) {
down_remove_blank();
down();
if (isFull())
return true;
else {
addNumber();
}
}
return false;
} public void setScore(TextView score) {
this.score = score;
} public void setBest(TextView best) {
this.best = best;
best.setText(getBestScore()+"");
}
public int getScore(){
return Integer.valueOf(score.getText().toString());
}
public int getBestScore(){
SharedPreferences sp = best.getContext().getSharedPreferences("bestScore.txt", Context.MODE_PRIVATE);
int bestScore=sp.getInt("bestScore", 0);
return bestScore;
}
public void setBestScore(int score){
SharedPreferences sp = best.getContext().getSharedPreferences("bestScore.txt", Context.MODE_PRIVATE);
Editor edit = sp.edit();
edit.putInt("bestScore", score);
edit.commit();
} }
<GridLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/root"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:columnCount="4"
android:rowCount="6"
tools:ignore="NewApi" > <TextView
android:layout_width="wrap_content"
android:layout_height="50dp"
android:layout_marginLeft="4dp"
android:layout_marginRight="4dp"
android:gravity="center"
android:text="Score:" /> <TextView
android:id="@+id/score"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:layout_marginLeft="4dp"
android:layout_marginRight="4dp"
android:gravity="center"
android:text="Score:" /> <TextView
android:layout_width="wrap_content"
android:layout_height="50dp"
android:gravity="center"
android:text="Best:" /> <TextView
android:id="@+id/best"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:gravity="center" /> <Button
android:layout_width="wrap_content"
android:layout_height="40dp"
android:layout_columnSpan="4"
android:onClick="reset"
android:text="reset" /> </GridLayout>

转载请注明出处:http://blog.csdn.net/liang5630/article/details/41251549

源代码及apk下载:

android 2048游戏实现的更多相关文章

  1. android 2048游戏、kotlin应用、跑马灯、动画源码

    Android精选源码 2048游戏源码 android实现获取号码归属地和其他信息诈骗.骚扰 android kotlin仿开眼app源码 android多种reveal动画效果 android K ...

  2. Android项目开发实战-2048游戏

    <2048>是一款比较流行的数字游戏,最早于2014年3月20日发行.原版2048首先在GitHub上发布,原作者是Gabriele Cirulli,后被移植到各个平台.这款游戏是基于&l ...

  3. Cocos2d-html5入门之2048游戏

    一.介绍 Cocos2d-JS是Cocos2d-x的Javascript版本,它的前身是Cocos2d-html5.在3.0版本以前叫做Cocos2d-html5,从3.0版本开始叫做Cocos2d- ...

  4. 2048游戏分析、讨论与扩展 - Part I - 游戏分析与讨论

    2048这个游戏从刚出開始就风靡整个世界. 本技术博客的目的是想对2048涉及到相关的全部问题进行仔细的分析与讨论,得到一些大家能够接受而且理解的结果. 在这基础上,扩展2048的游戏性,使其变得更好 ...

  5. 用javascript实现一个2048游戏

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

  6. Android原生游戏开发:使用JustWeEngine开发微信打飞机

    使用JustWeEngine开发微信打飞机: 作者博客: 博客园 引擎地址:JustWeEngine 示例代码:EngineDemo JustWeEngine? JustWeEngine是托管在Git ...

  7. powershell字符界面的,powershell加WPF界面的,2048游戏

    ------[序言]------ 1 2048游戏,有段时间很火,我在地铁上看有人玩过.没错,坐地铁很无聊,人家玩我就一直盯着看. 2 我在电脑上找了一个,试玩了以下,没几次格子就满了.我就气呼呼的放 ...

  8. [python] python实现2048游戏,及代码解析。

    我初学python,有不对之处望大家指教.转载请征得同意. 我在网络上也找了一些2048游戏代码的讲解,但都不是特别详细.所以我希望能够尽量详细的讲解.同时,有的地方我也不懂,希望大家能帮助补充.我会 ...

  9. 对弈类游戏的人工智能(5)--2048游戏AI的解读

    前言: 闲得没事, 网上搜"游戏AI", 看到一篇<<2048游戏的最佳算法是?来看看AI版作者的回答>>的文章. 而这篇文章刚好和之前讲的对弈类游戏AI对 ...

随机推荐

  1. 安卓、safari和微信各个浏览器的设计标准

  2. Vue-cli 3.0 构建项目

    Vue-cli是vue的一个脚手架,我们可以通过它来构建我们的前端项目 vue-cli3环境配置 //1. 安装nodeJS(已经集成npm) 首先需要安装node环境,可以直接到中文官网http:/ ...

  3. 1044 - Access denied for user 'root'@'%' to database 'xahy-blog' 解决方案二

    检查 user 表中'root'@'%' 的grant的权限 select HOST,USER,Grant_priv,Super_priv from mysql.`user`; 可以看到现在这两个权限 ...

  4. linux下mysql 查看默认端口号与修改端口号方法

    一.查看默认端口号 1.登录mysql [root@localhost ~]# mysql -uroot -pEnter password: 输入数据库密码: 2.使用show global vari ...

  5. Map的5种遍历方式

    public class MapF { public static void main(String[] args) { Map<String, Integer> tempMap = ne ...

  6. DataTable相关操作,筛选,取前N条数据,去重复行,获取指定列数据

    #region DataTable筛选,排序返回符合条件行组成的新DataTable或直接用DefaultView按条件返回      /// <summary>      /// Dat ...

  7. virtual box虚拟机在linux下设置共享文件夹

    使用的虚拟机版本是:VirtualBox-5.2.8-121009 使用的linux版本是:Ubuntu 12.04.5 LTS 和 Linux Mint 19 Tara 1. 安装增强功能包(Gue ...

  8. C++ vector基本用法

    转自金河http://www.cnblogs.com/wang7/archive/2012/04/27/2474138.html 1 基本操作 (1)头文件#include<vector> ...

  9. 紫书 例题8-4 UVa 11134(问题分解 + 贪心)

     这道题目可以把问题分解, 因为x坐标和y坐标的答案之间没有联系, 所以可以单独求两个坐标的答案 我一开始想的是按照左区间从小到大, 相同的时候从右区间从小到大排序, 然后WA 去uDebug找了数据 ...

  10. 时间就是金钱HNCOI2000(最短路)

    时间就是金钱HNCOI2000 版权声明:本篇随笔版权归作者YJSheep(www.cnblogs.com/yangyaojia)所有,转载请保留原地址! 人们总是选时间最短或费用最低的路线 例如, ...