android 五子棋开发
两天完成基本功能,再对其进行细节bug优化,本小白的思路。
思路:
1.用canvas绘制棋盘:得到手机的分辨率。棋盘大小为19*19。将手机宽屏分为21份,取中间19份为棋盘。上下空白位置为按钮功能。
如下:画出棋盘。
protected void onDraw(Canvas canvas,Paint paint) { canvas.drawColor(Color.rgb(128,64,0));
//canvas.drawColor(Color.WHITE);
w = chess_W/20;//平均分为21块 横坐标
h =scrren_h/2-chess_W/2;
for(int i =1; i<=19; i++ ){ //21分,画其中的地1份和第19份,0和20不画
canvas.drawLine(i*w,h+w,i*w,h+19*w, paint);
}//绘制直线
for(int i =1; i<=19;i++)
{ canvas.drawLine(w,h+i*w,19*w,h+i*w, paint);
}
paint.setStrokeWidth(3);
canvas.drawLine(w/2,h+w/2,w/2, h+19*w+w/2, paint);
canvas.drawLine(19*w+w/2,h+w/2,19*w+w/2,h+19*w+w/2, paint);
canvas.drawLine(w/2,h+w/2,19*w+w/2, h+w-w/2, paint);
canvas.drawLine(w/2,h+19*w+w/2,19*w+w/2, h+19*w+w/2, paint); /*
* (w,h+w) (19*w,h+w)
*
*
* (w,h+19*w) (19*w,h+19*w)
* */ //绘制四角
// 绘制这个三角形,你可以绘制任意多边形
paint.setColor(Color.rgb(227,207,0));
Path path = new Path();
path.moveTo(w/2,h+18*w);// 此点为多边形的起点
path.lineTo(w/2,h+19*w+w/2);
path.lineTo(2*w,h+19*w+w/2);
path.close(); // 使这些点构成封闭的多边形
canvas.drawPath(path, paint); Path path1 = new Path();
path1.moveTo(w/2,h+w/2);// 此点为多边形的起点
path1.lineTo(w/2,h+w+w);
path1.lineTo(2*w,h+w/2);
path1.close(); // 使这些点构成封闭的多边形
canvas.drawPath(path1, paint);
Path path2= new Path();
path2.moveTo(19*w+w/2,h+w/2);// 此点为多边形的起点
path2.lineTo(19*w+w/2,h+2*w);
path2.lineTo(18*w,h+w/2);
path2.close(); // 使这些点构成封闭的多边形
canvas.drawPath(path2, paint);
Path path3 = new Path();
path3.moveTo(19*w+w/2,h+19*w+w/2);// 此点为多边形的起点
path3.lineTo(18*w,h+19*w+w/2);
path3.lineTo(19*w+w/2,h+18*w);
path3.close(); // 使这些点构成封闭的多边形
canvas.drawPath(path3, paint);
}
2.在棋盘上落子:用canvas.bitmap。
需要注意的是:1.落子的地方以前没有被落子
2.落子的要保证落在棋盘的线上
3.落子要保证随心,不能与下棋者按下的棋点有偏差。
4.持黑先行,用步数控制应该下黑白棋。
5.采用一个二维数组存储数据,未落子为0,黑子为1,白子为2.
6.貌似canvas画图需要每次落子都要重新绘图,要从数组的提取数据依次绘图。可以自己设计算法,提高速度,比如根据步数,当发现棋子数量等于步数 后面就不用遍历了。最好不要全部遍历数组。
7.采用数组要注意数组越界问题。
8.边界问题!!!这个很重要。
落子落在棋线上和存储落子数据:
/判断落子
public float downx(float x,float y) { if (x > ChessDraw.w / 2 && x < ChessDraw.w * 19 + ChessDraw.w / 2) {
if (y > ChessDraw.h + ChessDraw.w / 2 && y < ChessDraw.h + 19 * ChessDraw.w + ChessDraw.w / 2) { int kx = (int) (x / ChessDraw.w);
int jx = (int) (x % ChessDraw.w); if (jx >= ChessDraw.w / 2) { //过了格子的一半
kx++;
x = kx * ChessDraw.w;
if(kx>20 || kx<=0) { //防止数组越界
x=-100;
}
}
else{ //未过格子的一半 x = kx * ChessDraw.w;
if(kx>20 || kx<=0) { //防止数组越界
x=-100;
}
} //获取落子的地点x
chess_x=kx;
}
else {
MySurfaceView.downQ = false; //不在棋盘,不允许落子
} }
else {
MySurfaceView.downQ = false; //不在棋盘,不允许落子
}
return x;
}
//判断落子 public float downy(float x, float y) { if (x > ChessDraw.w / 2 && x < ChessDraw.w * 19 + ChessDraw.w / 2) {
if (y > ChessDraw.h + ChessDraw.w / 2 && y < ChessDraw.h + 19 * ChessDraw.w + ChessDraw.w / 2) { //error
int ky = (int) ((y - ChessDraw.w / 2 - ChessDraw.h) / ChessDraw.w);
int jy = (int) ((y - ChessDraw.w / 2 - ChessDraw.h) % ChessDraw.w);
if (jy >= ChessDraw.w / 2) { //如果过了格子的一般
ky++;
y = ky * ChessDraw.w + ChessDraw.h ;
if(ky>20 || ky<=0) { //防止数组越界
y=-100;
} }
else{//如果没有过格子的一半 y=ky * ChessDraw.w + ChessDraw.h;
if(ky>20 || ky<=0) { //防止数组越界
y=-100;
}
}
//获取落子的地点x
chess_y=ky; }
else {
MySurfaceView.downQ = false;
}
}
else {
MySurfaceView.downQ = false; //不在棋盘,不允许落子
}
return y; }
public void downarr(int x, int y,int step){
if(step % 2 != 0)
chess_arr[x][y]=MySurfaceView.BLACK_CHESS;
else
chess_arr[x][y]=MySurfaceView.WRITE_CHESS;
}
3.胜利的判断,一共四个方向。对落子位子四个方向进行判断,如果一个方向累计成功五次以上即可。要注意数组坐标越界,可使用try catch,可优化算法。比如,前9步无需判断等,此处尚未优化,很暴力。
public ChessVictory(){
chessRule = new ChessRule();
}
public void chessvictory(int x, int y) {
int c =chessRule.chess_arr[x][y];
for (int i = 0; i < 4; i++) { //四个方向
linetrue2 = true;
linetrue1 = true;
victory_num=0;
tx1=0;
ty1=0;
tx2=0;
ty2=0;
for (int k = 0; k < 5; k++) {
try {
if (i == 0 && victory == false) {//对Y轴判断
ty1++;
int Ky1 = y+ty1;
ty2--;
int Ky2 = y+ty2;
if (linetrue1 && (chessRule.chess_arr[x][Ky1] == c)) { victory_num++;
} else {
linetrue1 = false;
}
//y轴--判断,注意数组越界
if (linetrue2 && chessRule.chess_arr[x][Ky2] ==c) { victory_num++;
} else {
linetrue2 = false;
}
if (victory_num >= 4) {
victory = true;
} }
//对x轴判断 if (i == 1 && victory == false) {
tx1++;
int Kx1 = x+tx1;
tx2--;
int Kx2 = x+tx2;
if (linetrue1 && chessRule.chess_arr[Kx1][y] == c) {
victory_num++;
} else {
linetrue1 = false;
}
//y轴--判断,注意数组越界
if (linetrue2 && chessRule.chess_arr[Kx2][y] ==c) {
victory_num++;
} else {
linetrue2 = false;
}
if (victory_num >= 4) {
victory = true; } }
//斜对角判断
if (i == 2 && victory == false) {
tx1++;
int kx1 = x+tx1;
tx2--;
int kx2 = x+tx2;
ty1++;
int ky1 = y+ty1;
ty2--;
int ky2 = y+ty2;
if (linetrue1 && chessRule.chess_arr[kx2][ky1] == c) {
victory_num++;
} else {
linetrue1 = false;
}
//y轴--判断,注意数组越界
if (linetrue2 && chessRule.chess_arr[kx1][ky2] == c) {
victory_num++;
} else {
linetrue2 = false;
}
if (victory_num >= 4) {
victory = true; }
}
//斜对角判断
if (i == 3 && victory == false) {
tx1++;
int kx1 = x+tx1;
tx2--;
int kx2 = x+tx2;
ty1++;
int ky1 = y+ty1;
ty2--;
int ky2 = y+ty2;
if (linetrue1 && chessRule.chess_arr[kx2][ky2] == c) {
victory_num++;
} else {
linetrue1 = false;
}
//y轴--判断,注意数组越界
if (linetrue2 && chessRule.chess_arr[kx1][ky1] ==c) {
victory_num++;
} else {
linetrue2 = false;
}
if (victory_num >= 4) {
victory = true; }
} } catch (Exception e) {
}
}
}
4.悔棋问题:
若只允许悔棋一次,可以直接对存储数据的数组操作。
若允许多次,可以在new一个数组进行每一步的存储(貌似数组存储不太好,具体实现的时候要考虑),悔棋时从此数组提取位子,在存储数据的数组上修改。
若要保存、载入等等。均可以在上次new的数组上进行保存。这样可以实现打谱。每次下一子,重现对局(未实现,只是思路)。
5.细节优化,比如从游戏进入后台,再从后台打开要保存原来的位置。本以为很难,其实在完成的时候只不过一笔带过。以及其余优化。
6.对战和AI设计。。。。继续完成。
android 五子棋开发的更多相关文章
- Android N开发 你需要知道的一切
title: Android N开发 你需要知道的一切 tags: Android N,Android7.0,Android --- 转载请注明出处:http://www.cnblogs.com/yi ...
- Android游戏开发实践(1)之NDK与JNI开发03
Android游戏开发实践(1)之NDK与JNI开发03 前面已经分享了两篇有关Android平台NDK与JNI开发相关的内容.以下列举前面两篇的链接地址,感兴趣的可以再回顾下.那么,这篇继续这个小专 ...
- Android游戏开发实践(1)之NDK与JNI开发01
Android游戏开发实践(1)之NDK与JNI开发01 NDK是Native Developement Kit的缩写,顾名思义,NDK是Google提供的一套原生Java代码与本地C/C++代码&q ...
- Android游戏开发实践(1)之NDK与JNI开发02
Android游戏开发实践(1)之NDK与JNI开发02 承接上篇Android游戏开发实践(1)之NDK与JNI开发01分享完JNI的基础和简要开发流程之后,再来分享下在Android环境下的JNI ...
- 【转】Android 底层开发的几点
我干了3年Android sdk开发,觉得到了瓶劲没法更进一步,于是花了一年多点时间,大概摸到点门径.根据前辈的经验,Android底层完全入门需要两年. 先说下我的入门过程:第零步,下载源码,我下的 ...
- 《Android NFC 开发实战详解 》简介+源码+样章+勘误ING
<Android NFC 开发实战详解>简介+源码+样章+勘误ING SkySeraph Mar. 14th 2014 Email:skyseraph00@163.com 更多精彩请直接 ...
- Android安全开发之安全使用HTTPS
Android安全开发之安全使用HTTPS 1.HTTPS简介 阿里聚安全的应用漏洞扫描器中有证书弱校验.主机名弱校验.webview未校验证书的检测项,这些检测项是针对APP采用HTTPS通信时容易 ...
- Android安全开发之通用签名风险
Android安全开发之通用签名风险 作者:伊樵.舟海.呆狐@阿里聚安全 1 通用签名风险简介 1.1 Android应用签名机制 阿里聚安全漏洞扫描器有一项检测服务是检测APP的通用签名风险.And ...
- Android安全开发之浅谈密钥硬编码
Android安全开发之浅谈密钥硬编码 作者:伊樵.呆狐@阿里聚安全 1 简介 在阿里聚安全的漏洞扫描器中和人工APP安全审计中,经常发现有开发者将密钥硬编码在Java代码.文件中,这样做会引起很大风 ...
随机推荐
- Java之Comparable接口和Comparator接口
Comparable & Comparator 都是用来实现集合中元素的比较.排序的: Comparable 是在集合内部定义的方法实现的排序: Comparator 是在集合外部实现的排序: ...
- 函数buf_LRU_search_and_free_block
/******************************************************************//** Try to free a replaceable bl ...
- poj3321
树映射到树状数组上 非常好的题目,给了我很多启发 题目要求动态求一个棵子树的节点个数 不禁联想到了前缀和,只要我们能用一个合适的优先级表示每个顶点,那么就好做了 我们可以考虑将子树表示成区间的形式 这 ...
- 5个难以置信的VS 2015预览版新特性
Visual Studio 2015 Preview包含了很多强大的新特性,无论你是从事WEB应用程序开发,还是桌面应用程序开发,甚至是移动应用开发,VS 2015都将大大提高你的开发效率.有几个特性 ...
- NOI2009植物大战僵尸
这题应该分两步来做: 1.拓扑排序,去掉无敌点 2.求最大闭合子图 需要注意几点: 1.拓扑排序时,如果(i,j)可以攻击到(x,y),那么增加(x,y)的入度,而不是(i,j)的入度 因为入度代表着 ...
- $http.post发的数据,后台取不到两种解决方案
方案一: var url = 'Gulugulus/setMenu', data = { menu: JSON.stringify(menu), test: 'a String' }, transFn ...
- 【转】Android动态改变对 onCreateDialog话框值 -- 不错不错!!!
原文网址:http://www.111cn.net/sj/android/46484.htm 使用方法是这样的,Activity.showDialog()激发Activity.onCreateDial ...
- Ruby类,模块1
类的扩展和继承 class Fixnum def dosome(str) puts str end def abs puts "覆盖了原有的方法" end end puts 1.c ...
- WCF发布后远程访问的域名解析问题
环境: VS2010 sp1,.net framework 4.0,windows server 2003 x64 ,iis 6.0 症状: WCF开发测试,本地调用都正常.发布后,在浏览器中访问ht ...
- java实现多继承
方法: 接口+组合 理由:通过接口实现客户端的使用时多继承类的多类, 通过组合实现客户端内部类的实现相关功能(而且有些共用的功能可以不总是多次实现). public interface GMapOb ...