多的不说,先上注释好的代码
主类:package 连连看V02;
import java.awt.Color; import java.awt.Graphics; import java.util.ArrayList; import javax.swing.ImageIcon; import javax.swing.JFrame; /* * * 连连看程序1.0 * 主类 * 本版本是从简单黑白棋版本过渡而来 */ public class ChessUI extends JFrame implements Config{ static{ //创建一个随机数d对象 java.util.Random random=new java.util.Random(); //将随机数加到一个队列中去 ArrayList<Integer> list=new ArrayList<Integer>(); for(int i=0;i<CHESSES.length*CHESSES[0].length/2;i++){ //随机一个数字,在1到10范围内 int number=random.nextInt(10)+1; //向队列中加两次,保证每个数字出现的次数为偶数 list.add(number); list.add(number); } /* * 创建图片时的理解: * 要求:1.图片总数比较是偶数,每一张图片的个数也必须是偶数 * 操作:将每一个类型的图片对应到队列的每一个数,CHESSES[i][j]表示的是在图形中的位置,CHESSES * 二维数组可以指向一个数,通过这个数,然后通过CHESSES_image[][]数组将图片按顺序存进去, * 这个1到10的随机数起到的是一个中间的过渡作用 */ for(int i=0;i<CHESSES.length;i++){ for(int j=0;j<CHESSES[i].length;j++){ //取一个随机的队列中的下标 int index=random.nextInt(list.size()); int t=list.remove(index);//从队列中移除这个随机数 //对二维数组中这个位置就行赋值,下一次循环时,代表这个t值的CHESSES[i][j]就取不到了, //因为这个t值已经从list中除去了 CHESSES[i][j]=t; CHESSES_image[i][j]=new ImageIcon("src/我的连连看V01/image/"+t+".gif"); } } } public static void main(String[] args){ ChessUI UI=new ChessUI(); UI.initChessUI(); } /* * 初始化界面的方法 */ public void initChessUI(){ this.setTitle("连连看v01"); this.setLocation(200, 200); this.setSize(600,600); //设置背景色 this.getContentPane().setBackground(new Color(134,224,13)); this.setResizable(false);//设置不可更改大小 this.setDefaultCloseOperation(3); this.setVisible(true); //setVisible之后就行抓取画布 Graphics g=this.getGraphics(); //给画布加上监听器 ChessListener csl=new ChessListener(g,this); //这里直接写UI是不行的,因为UI只是主类中的,并不能在方法中调用,所以用this表示ChessUI的对象 this.addMouseListener(csl); } /* * 编写一个重绘的方法 * * */ public void paint(Graphics g){ super.paint(g); this.paintTable(g); this.paintImage(g); } //重写一个画棋盘的方法 public void paintTable(Graphics g){ //画棋盘 //画横线 for(int i=0;i<ROWS;i++){ g.drawLine(X0, Y0+SIZE*i, X0+SIZE*(COLUMNS-1), Y0+SIZE*i); } //画竖线 for(int j=0;j<COLUMNS;j++){ g.drawLine(X0+SIZE*j,Y0,X0+SIZE*j,Y0+SIZE*(COLUMNS-1)); } } //重写一个画图片,其实也相当于棋子的方法 public void paintImage(Graphics g){ //重绘图片 for(int i=0;i<CHESSES.length;i++){ for(int j=0;j<CHESSES[i].length;j++){ //根据下标计算图片左上角的坐标 int X1=X0+SIZE*j; int Y1=Y0+SIZE*i; //如果数组值不等于0,就进行重绘 if(CHESSES[i][j]!=0){ g.drawImage(CHESSES_image[i][j].getImage(), X1, Y1, SIZE, SIZE, null); } } } } }
监听器类:
package 连连看V02; import java.awt.Graphics; import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; /* * 监听器 * * * * */ public class ChessListener extends MouseAdapter implements Config{ private Graphics g; private ChessUI UI; int count=0;//用一个计数器使每两次相邻的点击在一个"组"内,再进行判断 int r1,c1,r2,c2;//两次点击的下标 int number1,number2; //构造函数,传入Graphics g这个变量 public ChessListener(Graphics g,ChessUI UI){ this.g=g; this.UI=UI; } //监听鼠标点击的地方,就行放子 //count的作用就是将这两次相邻的点击分为一组,避免了第二次点击的图形会和第三次点击的进行判断 public void mouseReleased(MouseEvent e){ int x=e.getX(); int y=e.getY(); if(count==0){ //算出是图片在在哪个单元格内 c1=(x-X0)/SIZE; r1=(y-Y0)/SIZE; number1=CHESSES[r1][c1]; count++; }else{ int c2=(x-X0)/SIZE; int r2=(y-Y0)/SIZE; number2=CHESSES[r2][c2]; count--; //如果是相同的图片,不是自己本身,而且不为空,则再进行算法的判断 if(number1==number2&(r1!= r2 || c1 != c2)&number1!=0){ //算法只要满足一个,就可以消去,消去就将对应位置的数组数据变为0 if(Suanfa.checkRow(r1,c1,r2,c2)||Suanfa.checkCulumn(r1,c1,r2,c2)){ CHESSES[r1][c1]=0; CHESSES[r2][c2]=0; // 刷新窗体,是个终极办法?但是要传入UI对象,所以要通过Listener()括号内传入UI javax.swing.SwingUtilities.updateComponentTreeUI(UI); } } } } }
算法类:
package 连连看V02; /* * 算法类:判断两个图片是否会被消掉 * 有以下几种情况: * 1.同行;2同列; * 3.两个图片只用2条线连起来(4种情况): * 4.两个图片用3条线连起来(这个是最复杂的,8种情况): * * * * */ public class Suanfa implements Config{ //同行的情况,算法 public static boolean checkRow(int r1,int c1,int r2,int c2){ //如果不同行,则返回false if(r1!=r2){ return false; } //同行时,那么这两个图片中间的都必须为空,否则返回false for(int i=Math.min(c1, c2)+1;i<Math.max(c1, c2);i++){ if(CHESSES[r1][i]!=0){ return false; } } //如果上面都成立则返回true return true; } //同列的情况,算法 public static boolean checkCulumn(int r1,int c1,int r2,int c2){ //同上 if(c1!=c2){ return false; } //同列时,两个图片的中间必须为空,否则返回false for(int j=Math.min(r1, r2)+1;j<Math.max(r1, r2);j++){ if(CHESSES[j][c1]!=0){ return false; } } //如果上面都成立则返回true return true; } }
保存常用参数的接口:package 连连看V02;
import javax.swing.ImageIcon; /* * 这是初级版本,画线只是为了方便标记位置 * 到后面的版本会去掉的 * * * */ //定义为接口,里面的属性可以被全局使用 public interface Config { public final int X0=50,Y0=50;//左上角的那个点的坐标 public final int ROWS=9,COLUMNS = 9;//定义横线和竖线的条数都为9 public final int SIZE=60;//定义单元格的大小为60 public final int CHESSSIZE=50;//定义棋子大小为50 //设置标记“棋子”的数组 public int[][] CHESSES=new int[ROWS-1][COLUMNS-1]; //设置放置图片的数组,与“棋子”位置对应 public ImageIcon[][] CHESSES_image=new ImageIcon[ROWS-1][COLUMNS-1]; }
这是代码,注释都在里面,今天学到了一个比较实用的自我检测错误的方法:打印。刚开始写完第一编。发现图片可以显示,但是点击鼠标无反应。这样我就在监听器类中就行打印。最后发现在将对应的数组赋值为0的操作中没有打印,说明那个if条件判断出了点问题,if条件判断里面是算法中的方法,找到根源,果然发现是在for循环中,i,j的下限是从两个数最小值+1开始的,因为忘记+1,所以导致无法判断。
下一步将对连连看进行改进吧:
1.背景换个好看点的。
2.算法写好,有许多种情况。
3.在鼠标按下的时候图片需要换一种状态,加个边框或者其他,今天在讲导入图片时,我记得有一个方法在后面是可以加上鼠标点击后的图片的。可以查下
上面是比较容易修改的。下面写几个扩展的,难一点吧
4.在图形消去的同时,画出比较淡的线连起这两个图片(初步想法觉得是在消去图形的前面调用drawLine的方法,具体要画什么样的线,当然可以根据上面判断的到底是调用的哪个算法来进行画线) 这里面还存在一个问题,有些情况会同时满足几个算法,例如2个在一行的,并且中间没有图形的,需要一个优先选择,选择那个最简单的来画线(想了下,在如果可以想消之后,套用3个if,就可以解决优先级顺序问题)
if(最简单的成立){
用最简单的算法
}else if(次简单的成立){
用次简单的方法
}else if{
用最麻烦的方法
}
5.消去的同时加个声音? 呵呵
6.提示和自动消除,重排序,当没有可消的时候,进行处理(1.直接随机换位置。2.提示不可消,自己手动用重排序功能)
7.倒计时,更高级的就是如果自己连的比较好,那么时间可以往上加。
8.不一定是一出来就是矩形,可以各种图形,像QQ连连看,随机的
9.是否可以弄个登陆界面,有几个选项,选择游戏难度,是否有声音,还有排行榜,或者可以选择用什么样的图形去排列图片
扩展功能还可以有很多吧,不想再多说了。因为我QQ连连看也有7W多分。所以对这个功能的多样性还是比较了解的。就看能实现多少了。慢慢来吧