以下每行代码,文字均为原创,转载请注明出处.

程序一共分为7个文件,每个文件为一个类

文件名 功能描述
Test.java 测试类,包含main()函数
Mine.java 设计主界面,
Calmine.java 随机雷的位置.计算雷区点击后应该显示的数字
My_button.java 继承自button类,添加按钮的坐标x,y.
Num_Mine.java 雷数,包括用户以标记的雷数,标记正确的雷数,以及总雷数
ClickLone.java 鼠标左击处理
ClickRone.java 鼠标右击处理


Test.java

测试类,Mine()接受的三个参数分别为雷区的长,宽,总雷数

package mine_sweep;
public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		new Mine(16,16,20);
	}

}

Mine.java

继承自JFame类,采用BorderLayout()布局,.NORTH区域放置显示面板JPanel,提示用户以标记的雷数,.SOUTH区域放置雷区,采用GridLayout布局

用一个双重循环向雷区minepanel添加按钮,同时在buttons[]中储存该按钮的引用.

同时添加按钮监听器..在用户点击了右键后,负责更新标记的类数


package mine_sweep;
import java.awt.*;
import java.awt.event.*;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextArea;


@SuppressWarnings("serial")
public class Mine extends JFrame {

	Integer rn;				//用户已标记的雷数
	Integer sn;				//标记对的雷数
	public Mine(int rows,int cols,int n)		//雷区的行数,列数,雷数
	{
		My_button buttons[]=new My_button[rows*cols];	//记录每个按钮的引用
		Integer mine[]= new Integer[rows*cols];			//每个位置的数字
		Boolean ismine[]=new Boolean[rows*cols];		//是否有雷的标志
		new Calmine(mine,ismine,rows,cols,n);
		this.setSize(500, 450);
		this.setTitle("mine_sweep");
		this.setLayout(new BorderLayout());		

		rn=new Integer(0);				//用户标记的雷数
		sn=new Integer(0);				//标记对的雷数
		JPanel showpanel=new JPanel();			//设置控制面板
		showpanel.setLayout(new FlowLayout(FlowLayout.LEFT));
		JTextArea showArea=new JTextArea("sum of mine: "+n);
		JTextArea showrmArea=new JTextArea("num of marked  mine: "+rn.intValue());
		showArea.setEditable(false);
		showrmArea.setEditable(false);
		showpanel.add(showrmArea );
		showpanel.add(showArea);
		this.add(showpanel,BorderLayout.NORTH);  	
		
		JPanel minepanel=new JPanel();			//设置雷区
		minepanel.setLayout(new GridLayout(rows,cols));
		minepanel.setSize(500, 450);
		Mine t=this;
		for(int i=0;i<rows;i++)
		{
			for(int j=0;j<cols;j++)
			{
				My_button btn=new My_button(i,j);
				buttons[i*cols+j]=btn;
				minepanel.add(buttons[i*cols+j] );
				btn.setBackground(Color.blue);
				btn.addMouseListener(new MouseAdapter()
				{
					public void mouseClicked(MouseEvent e)
					{
						if(e.getButton()==MouseEvent.BUTTON1)
						{
							int ans=rn.intValue();
							if(btn.getLabel()!="#")
								 ans = new ClickLone(buttons,btn.x,btn.y,rows,cols,mine,ismine,t,rn).rep();		
							showrmArea.setText("num of marked  mine: "+ans);	//更新已标记雷数;
							rn=new Integer(ans);
						}
						if(e.getButton()==MouseEvent.BUTTON3)
						{
							Num_Mine num_mine=new Num_Mine(rn,sn,n);
							num_mine=new ClickRone(buttons,btn.x,btn.y,rows,cols,num_mine,t,ismine).repnum();
							rn=new Integer(num_mine.mark_mine);
							showrmArea.setText("num of marked  mine: "+rn);	//更新已标记雷数
							sn=new Integer(num_mine.mark_correct_mine);	//更新标记正确雷数
							
						}
					}
					
				}); //添加鼠标点击事件
			}
			
			
		}
		this.addWindowListener(new WindowAdapter(){		//添加窗口关闭监听器
			public void windowClosing(WindowEvent e)
			{
				t.dispose();
			}
		});
		this.add(minepanel,BorderLayout.SOUTH);
		minepanel.setVisible(true);
		this.setVisible(true);
	}

}

Calmine.java

随机雷的位置.储存在ismine[]中. 并通过dx[].dy[]遍历一个按钮周围的8个位置.计算出每个按钮点击后应该显示的值,计算出的结果存储在mine[]中

注意:随机数可能产生重复的,导致雷数不够.所以,要加一个while循环放置产生重复的位置

package mine_sweep;

import java.util.Random;

public class Calmine {
	Integer mine[];
	Boolean ismine[];
	int rows,cols;
	int dx[]={-1,1,0};
	int dy[]={-1,1,0};
	int num;
	Random rd;
	Calmine(Integer m[],Boolean ism[],int r,int l,int n)
	{
		rows=r;
		cols=l;
		num=n;
		mine=m;
		ismine=ism;
		rd=new Random();
		for(int i=0;i<rows*cols;i++)			//初始化
		{
			ismine[i]=new Boolean(false);
			mine[i]=new Integer(0);
		}
		for(int i=0;i<num;i++)		//随机产生n雷
		{
			int rom=rd.nextInt(rows*cols);
			while(ismine[rom]==true)			//防止生成重复的随机数
				rom=rd.nextInt(rows*cols);
			ismine[rom]=new Boolean(true);
		}
		for(int i=0;i<rows;i++)			//计算每个按钮按下后应该显示的数字
			for(int j=0;j<cols;j++)
			{
				int t=0;
				for(int x=0;x<3;x++)		//周围8个位置
					for(int y=0;y<3;y++)
					{
						int nx=dx[x]+i,ny=dy[y]+j;
						if(nx>=0&&ny>=0&&nx<rows&&ny<cols&&(!ismine[i*cols+j].booleanValue()))
						{
							t+=ismine[nx*cols+ny].booleanValue()?1:0;
						}
					}
				mine[i*cols+j]=new Integer(t);
			}
	}

}

My_button.java

继承自Button类,添加x,y.标记该按钮是处以雷区的的第几行第几列

package mine_sweep;

import java.awt.Button;

 @SuppressWarnings("serial")
public class My_button extends Button {
	 int x;
	 int y;
	 My_button()
	 {
		 super();
	 }
	 My_button(int a,int b)
	 {
		 super();
		 x=a;
		 y=b;
	 }
}

Num_Mine.java

存储用户以标记的雷数,标记正确的雷数,以及雷区的总雷数

package mine_sweep;

public class Num_Mine {
	Integer mark_mine;
	Integer mark_correct_mine;
	Integer sum_mine;
	Num_Mine(Integer rn,Integer sn,Integer n)
	{
		mark_mine=new Integer(rn);
		mark_correct_mine=new Integer(sn);
		sum_mine=new Integer(n);

	}

}

ClickLone.java

左击事件的处理. 分两种情况:

     1,点击的位置是雷,这时,弹出一个模态对话框显示GAME OVER,结束游戏

     2.点击的地方不是雷,这时,需要把该按钮设置为setEnabled(false).并通过mine[]数组中预先计算出应该显示的数字设置按钮标签,如果为零,这设置为空格..然后,如果点击的位置对应的mine[]的值为0,则调用BFS,不断向周围扩散,直到遇见数字或雷为止(广度优先搜索,队列)

package mine_sweep;

import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.util.LinkedList;
import java.util.Queue;

import javax.swing.JButton;
import javax.swing.JDialog;

public class ClickLone {
	My_button buttons[]; 		//按钮的引用
	int i,j;
	int rows,cols;
	Integer  mine[];
	Boolean ismine[];
	boolean used[];
	Integer rn;				//已经标记的雷数
	Mine sjm;
	int dx[]={1,-1,0};
	int dy[]={1,-1,0};
	ClickLone(My_button[] tem,int a,int b,int r,int l,Integer mines[],Boolean ismines[],Mine Jm,Integer n)
	{
		buttons=tem;
		rn=n;
		mine=mines;
		ismine=ismines;
		rows=r;
		cols=l;
		i=a;
		j=b;
		sjm=Jm;
		used=new boolean[rows*cols];
		for(int i=0;i<rows*cols;i++)
			used[i]=false;
		if(ismine[i*cols+j].booleanValue())			//如果选中地雷
		{
			buttons[i*cols+j].setEnabled(false);
			buttons[i*cols+j].setLabel("*");
			buttons[i*cols+j].setBackground(Color.red);
			JButton btc=new JButton("GAME OVER!");
			JDialog dialog=new JDialog(sjm,"game_over");
			dialog.setLayout(new FlowLayout());
			dialog.addWindowListener(new WindowAdapter()		//对话框监听器
			{

				@Override
				public void windowClosing(
						WindowEvent e) {
					// TODO Auto-generated method stub
					super.windowClosing(e);
					dialog.dispose();
					sjm.dispose();
				}
				
				
			});
			btc.addActionListener(new ActionListener(){

				@Override
				
				public void actionPerformed(
						ActionEvent e) {
					// TODO Auto-generated method stub
					dialog.dispose();
					sjm.dispose();
				}
				
			});
			dialog.add(btc);
			dialog.setModal(true);				//弹出game_over窗口
			dialog.setSize(220,150 );
			dialog.setLocation(350, 250);
			dialog.setVisible(true);
			
			return;
		}
		Bfs(i,j);
//		buttons[i*cols+j].setLabel(mine[i*cols+j].toString());
	//	if(ismine[i*cols+j].booleanValue())
	//	{
	//		buttons[i*cols+j].setLabel("*");
			
	//	}
		
	}
	int rep()
	{
		return rn.intValue();
	}
	void Bfs(int i,int j)			//广度优先搜索
	{
		Queue<Integer> que=new LinkedList<Integer>();
		que.offer(new Integer(i*cols+j));
		while(!que.isEmpty())
		{
			int t=que.poll().intValue();
			if(new String("#").equals(buttons[t].getLabel()))		//如果该位置标记为雷,说明用户标记错误
				rn=new Integer(rn.intValue()-1);			//用户标记的雷数-1
			if(new String("0").equals(mine[t].toString()) )
				buttons[t].setLabel(" ");
			else
			{
				buttons[t].setLabel(mine[t].toString());
			}

			buttons[t].setEnabled(false);
			buttons[t].setBackground(Color.gray);
			if(mine[t].intValue()!=0)
				continue;
			i=t/cols;
			j=t%cols;
			for(int x=0;x<3;x++)
				for(int y=0;y<3;y++)
				{
					int nx=i+dx[x],ny=j+dy[y];
					if(nx>=0&&ny>=0&&nx<rows&&ny<cols&&			//没有越界
						mine[i*cols+j].intValue()==0					// 是空白区域且没有雷
						&&(!ismine[nx*cols+ny].booleanValue())
						&&(!(nx==i&&ny==j))
						&&(!used[nx*cols+ny])
					)	
					{
						used[nx*cols+ny]=true;
						que.offer(new Integer(nx*cols+ny));
						
					}
				}

		}
		
	}
}

ClickRone.java

处理右键事件,如果该位置还没有被标记,点击后将按钮的标签设置为"#" ,如果已经被标记为"#" ,则将标记清除.

如果标记的雷数==标记对的雷数==总雷数,接结束游戏,弹出YOU WIN 对话框

package mine_sweep;

import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;

import javax.swing.JButton;
import javax.swing.JDialog;

public class ClickRone {
	My_button buttons[];
	int x,y;
	int rows,cols;
	Mine sjm;
	Boolean ismine[];
	Num_Mine num_mine;
	ClickRone(My_button button[],int a,int b,int r,int l,Num_Mine num,Mine jm,Boolean ism[])
	{
		ismine=ism;
		buttons=button;
		num_mine=num;
		sjm=jm;
		x=a;
		y=b;
		rows=r;
		cols=l;
		if(buttons[x*cols+y].getLabel()=="#")			//该位置以被标记为雷,取消标记
		{
			num_mine.mark_mine=new Integer(num_mine.mark_mine.intValue()-1);
			buttons[x*cols+y].setBackground(Color.blue);
			buttons[x*cols+y].setLabel(" ");
			if(ismine[x*cols+y].booleanValue())			//如果该位置是雷
				num_mine.mark_correct_mine=new Integer(num_mine.mark_correct_mine.intValue()-1);
		}
		else						//加上雷的标记	
		{
			num_mine.mark_mine=new Integer(num_mine.mark_mine.intValue()+1);
			buttons[x*cols+y].setBackground(Color.yellow);
			buttons[x*cols+y].setLabel("#");
			if(ismine[x*cols+y].booleanValue())		//如果标记正确
				num_mine.mark_correct_mine=new Integer(num_mine.mark_correct_mine.intValue()+1);
		}
		if(num_mine.mark_correct_mine.intValue()==num_mine.mark_mine.intValue()
				&&num_mine.mark_mine.intValue()==num_mine.sum_mine.intValue())
		{
			JButton btc=new JButton("YOU WIN");
			JDialog dialog=new JDialog(sjm,"you_win");
			dialog.setLayout(new FlowLayout());
			dialog.addWindowListener(new WindowAdapter()		//对话框监听器
			{

				@Override
				public void windowClosing(
						WindowEvent e) {
					// TODO Auto-generated method stub
					super.windowClosing(e);
					dialog.dispose();
					sjm.dispose();
				}
				
				
			});
			btc.addActionListener(new ActionListener(){			

				@Override
				public void actionPerformed(
						ActionEvent e) {
					// TODO Auto-generated method stub
					dialog.dispose();
					sjm.dispose();
				}
				
			});
			dialog.add(btc);
			dialog.setModal(true);				//弹出you_win窗口
			dialog.setSize(220,150 );
			dialog.setLocation(350, 250);
			dialog.setVisible(true);
		}
	}
	Num_Mine repnum()
	{
		return num_mine;
	}

}

下面是运行截图:

从拿到题目,到基本完成,一共用了整整一天的时间(早上八点~第二天凌晨1点).

后来用了3天改了十数个bug.终于做成了现在比较完善的程序.