`
剑&箫
  • 浏览: 54887 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
社区版块
存档分类
最新评论

以打砖块游戏来理解多线程

    博客分类:
  • Java
阅读更多
前几天学习了多线程,现在总结自己做的一个打砖块的游戏,以此来加深对多线程的理解(如有不正确的地方欢迎指正!)。首先来看游戏的效果图:




首先要有一个界面,界面的实现在前面已经作过很多次了,具体代码如下:
/**
* 初始化窗体
*/
public void initFrame(){
this.setTitle("喷怒的小球");//设置窗体的标题
this.setSize(500, 750);//设置窗体的大小
//this.getContentPane().setBackground(Color.BLACK);
this.setLayout(new FlowLayout());//设置流式布局管理器
JButton bt = new JButton("开始");
JButton bt1 = new JButton("停止");

JPanel panel = new JPanel ();

Dimension d = new Dimension(495,650);

panel.setBackground(Color.BLACK);
panel.setPreferredSize(d);

this.add(bt);
this.add(bt1);
this.add(panel);

this.setResizable(false);//设置窗体的大小不可变
this.setDefaultCloseOperation(3);//点击关闭时退出窗体
this.setVisible(true);//将窗体显示在屏幕上

//设置焦点
bt.setFocusable(false);
bt1.setFocusable(false);
panel.setFocusable(true);

final Graphics g = panel.getGraphics();//得到画布
}
得到窗体之后,需要一个挡板,所以定义一个挡板类,并且这个挡板能够在窗体底部水平移动,所以这里是定义的挡板类是实现MouseMotionListener的接口,使得画出的挡板能够随着鼠标的移动而移动,然后在这个类里面定义挡板的属性和实现画挡板的方法。挡板的属性有左上角的坐标,长,宽以及颜色等,为了美观,这里是直接画挡板的一张图片,具体代码如下所示:
public class Fender implements MouseMotionListener{

public static int x = 0;
public static int getX() {
return x;
}

public int y = 630;
public int width = 100;
private int height = 20;
private JPanel panel;
private Graphics g;

public Fender(){}
public Fender(netjava.wxh0807pm1.BallFrame.mypanel panel){

this.panel = panel;
g = panel.getGraphics();
}
//重写父类的方法
public void mouseMoved(MouseEvent e){
//清除图像
    g.setColor(panel.getBackground());
    g.fillRect(x, y, width, height);
x = e.getX();

//g.setColor(Color.RED);
if(x>=400){
x=400;
}
//画挡板
    createFender(g,x,y,width,height);
    }
//画挡板的方法
public void createFender(Graphics g,int x,int y,int width,int height){
javax.swing.ImageIcon icon = new javax.swing.ImageIcon("src\\netjava\\wxh0807pm1\\image\\5.png");
g.drawImage(icon.getImage(), x, y, width, height, null);
}
   
public void mouseDragged(MouseEvent e){

}
}
然后需要画出自己设计的砖块,这里是直接以地图的形式画砖块的。首先是准备几张砖块的图片,然后是把要画得区域看成一个二维数组,二维数组中的元素为零的地方表示该区域没有画砖块,以不同数字表示不同的砖块,然后在另外一个文件中设计二维数组的元素以画出自己想要画得地图。二维数组设计好之后,首先要定义一个方法来把文件读取到内存中,这时就用到了输入输出流的知识,在前面已经总结过,这里不再罗嗦了。但是读取到的是字符串,所以还需要定义一个方法将字符串转化为数组,然后还要定义一个得到图片的静态方法,最后要定义一个根据得到的数组和图片创建地图的方法。将前面三个方法写成一个类,具体代码如下所示:
public class MapTest {

/**
* 读取文件中的地图数据
*
* @param path
* @return
*/
public static int[][] readMap(String path) {
try {
// 创建文件输入流
FileInputStream fis = new FileInputStream(path);
BufferedInputStream bis = new BufferedInputStream(fis);

byte[] bs = new byte[bis.available()];
// 将数据从流中读取到数组中
bis.read(bs);

String str = new String(bs);

// 对字符串进行处理

// System.out.println(str);

int[][] arr = changeToArray(str);
return arr;

} catch (Exception ef) {
ef.printStackTrace();
}

return null;
}

/**
* 将字符串转化为数组
*
* @param str
* @return
*/
private static int[][] changeToArray(String str) {

// 根据回车换行符将字符串分割为字符串数组
String[] strs = str.split("\r\n");

int[][] array = new int[strs.length][strs[0].length()];

// 遍历字符串数组
for (int i = 0; i < strs.length; i++) {
String s = strs[i];
// 对字符串进行解析
char[] cs = s.toCharArray();

for (int j = 0; j < cs.length; j++) {
char c = cs[j];
// 将字符串转成数字
int num = Integer.parseInt(c + "");

array[i][j] = num;

}

}
return array;
}

//根据路径得到图片对象的方法
public static ImageIcon createImageIcon(String path) {
java.net.URL url = MapTest.class.getResource(path);
ImageIcon icon = new ImageIcon(url);
return icon;
}
}
最后一个方法的代码如下:
/**
* 根据地图数组创建地图
* @param array
*/
public static void createMap(int[][] array,Graphics g){

for(int i=0;i<array.length;i++){
for(int j=0;j<array[i].length;j++){

if(array[i][j]!=0){
int num = array[i][j];
String path = "image/"+num+".png";
//根据路径构造图片对象
ImageIcon icon = MapTest.createImageIcon(path);
g.drawImage(icon.getImage(), 35*j, 15*i, null);

}

}

}

}
上面的方法都写成之后只要调用就可以实现砖块的绘制了。现在还需要绘制一个小球,这个小球是一个线程,所以定义一个小球类,在该类里面定义小球的属性和画得方法,在小球的移动过程中还要判断小球与界面的左右以及上边的碰撞反弹以及小球与砖块的碰撞。小球与砖块的碰撞主要分别从砖块的四条边考虑与小球的碰撞,因为根据上面的方法得到数组可以得到每个砖块的位置,然后在遍历数组,判断数组中的每一个砖块是否与小球相撞,然后在做相应的反弹,砖块碰到小球之后要把砖块消掉,所谓消掉就是把砖块画成与背景一样的颜色,把数组中对应的元素变为零。然后在判断小球是否与挡板碰撞,如果碰撞,则弹回,如果挡板没有接住小球,则游戏结束。然后在写一个方法判断是否赢了,同样是遍历上面得到的数组,如果数组的元素全为零,则说明砖块全被打完了,则赢了。具体点的代码如下所示:
/**
* 小球类
* @author lenovo
*
*/
public class Ball extends Thread{

java.util.Random rd = new java.util.Random();

public static  int x0=240;
public static int  y0=605;
private int width=20;
private int height=20;
private int x1;
private int y1;

private JPanel panel;
private Graphics g;
private Fender fd;

public static boolean isStop=false;
public static boolean isPause=false;

public Ball(){}
public Ball(JPanel panel,Fender fd){
this.fd = fd;
this.panel = panel;
g = panel.getGraphics();


//小球的增量
x1 = 8;
y1 = -8;

}

public void run(){
draw();
}



public void draw(){
Fender fd = new Fender();
while(!isStop){

while(!isPause){

//javax.swing.ImageIcon icon = new javax.swing.ImageIcon("src\\netjava\\wxh0807pm1\\image\\6.png");

//清除图像
    g.setColor(panel.getBackground());
    g.fillRect(x0, y0, width, height);

//遍历数组,判断是否与砖块相撞
//int[][] array = MapTest.readMap("src\\netjava\\wxh0806\\image\\map");
for (int i=0;i<BallFrame.arr.length;i++){
for (int j=0;j<BallFrame.arr[i].length;j++){
if(BallFrame.arr[i][j]!=0){
if (x0>=35*j-1&&x0<=35*j+10&&y0<=15*i+15&&y0>=15*i){//砖块左边碰撞
g.setColor(panel.getBackground());
g.fillRect(35*j, 15*i, 36, 15);

//System.out.println("11");
BallFrame.arr[i][j] = 0;
x1=-x1;
}else if (y0>=15*i-1&&y0<=15*i+15&&x0<=35*j+35&&x0>=35*j){//砖块上边判断
g.setColor(panel.getBackground());
g.fillRect(35*j, 15*i, 36, 15);

//System.out.println("12");
BallFrame.arr[i][j]=0;
y1=-y1;
}else if (y0<=15*i+15+1&&y0>=15*i&&x0<=35*j+35&&x0>=35*j){//砖块下边判断
g.setColor(panel.getBackground());
g.fillRect(35*j, 15*i, 36, 15);

//System.out.println("13");
BallFrame.arr[i][j]=0;
y1=-y1;
}else if(x0>=35*j+35+1&&x0<=35*j+35-10&&y0<=15*i+15&&y0>=15*i){//砖块右边判断
g.setColor(panel.getBackground());
g.fillRect(35*j, 15*i, 36, 15);

//System.out.println("14");
BallFrame.arr[i][j]=0;
x1=-x1;
}
}


}
}


isWin(BallFrame.arr);

if (x1!=0){
if (x0<=0||x0>=470){//左右两壁
x1=-x1;
//System.out.println("1");
}else if (y0<=0){//上下两壁
y1=-y1;
//System.out.println("2");
}else if ((x0<=0&&y0<=0)||x0<=0||(x0>=470&&y0<=0)||x0>=470){//垂直碰撞四壁
x1=-x1;y1=-y1;
//System.out.println("3");


}

else if (y0>=630-20&&y0<=630-20+10&&x0<=Fender.getX()+100-10&&x0>=Fender.getX()-10){
//System.out.println("------------");
y1=-y1;
//System.out.println("0");
}else if (y0>640&&y0<650){
javax.swing.JOptionPane.showMessageDialog(null, "加油哦!");
}
x0+=x1;
y0+=y1;

//画球
createBall(g,x0,y0);
// g.drawImage(icon.getImage(), x0+=x1, y0+=y1, null);

// g.setColor(Color.RED);
// g.fillOval(x0+=x1,y0+=y1,width,height);

}

try{
Thread.sleep(40);
}catch(Exception ep){
ep.printStackTrace();
}


}

try{
Thread.sleep(1);
}catch(Exception ef){
ef.printStackTrace();
}



}

}

//画球的方法
public void createBall(Graphics g,int x,int y){
javax.swing.ImageIcon icon = new javax.swing.ImageIcon("src\\netjava\\wxh0807pm1\\image\\6.png");
g.drawImage(icon.getImage(), x, y, null);
}


/**
* 判断输赢的方法
* @param chars
*/
public void isWin(int[][]array){
int count=0;
for(int m=0;m<array.length;m++){
for(int n=0;n<array[m].length;n++){
if(array[m][n]!=0){
count++;
}
}
}
System.out.println(count);
if(count==0){
JOptionPane.showMessageDialog(null, "YOU WIN!!!");
stopThread();
}
}

}

然后再在小球类里面定义暂停、继续等的方法来控制小球的线程,具体代码如下所示:
//暂停的方法
public static void pauseThread(){
isPause=true;
}

//继续的方法
public static void resumeThread(){
isPause=false;
}

//停止的方法
public static void stopThread(){
isPause=true;
isStop=true;
}

//初始的方法
public static void initThread(){
isPause=false;
isStop=false;
}

然后再在初始化窗体的方法里面定义一个内部匿名类,来启动线程,但是在这个类里面用到的不在此类里的变量都要定义成final,具体代码如下:
//匿名内部类
ActionListener alt = new ActionListener(){


public void actionPerformed(ActionEvent e){

String command = e.getActionCommand();
if (command.equals("开始")){


//读取文件
int[][] array = MapTest.readMap("src\\netjava\\wxh0807pm1\\image\\map");

arr = array;

//画图片
createMap(array,g);

Ball b = new Ball(panel,fd);
b.start();

bt.setText("暂停");

}

if (command.equals("暂停")){

Ball.pauseThread();
bt.setText("继续");
}

if (command.equals("继续")){
Ball.resumeThread();
bt.setText("暂停");
}

if (command.equals("停止")){

Ball.stopThread();
bt.setText("开始");
}
}

};
//添加监听器
bt.addActionListener(alt);
bt1.addActionListener(alt);
Fender fd = new Fender(panel);
panel.addMouseMotionListener(fd);

然后再重绘挡板、小球以及砖块就可以了,重绘代码如下所示:
//重绘
class mypanel extends JPanel{

public void paint(Graphics g){

//重写父类的方法
super.paint(g);
//遍历砖块数组,实现重绘
for(int i=0;i<arr.length;i++){
for(int j=0;j<arr[i].length;j++){
if(arr[i][j]!=0){
int num=arr[i][j];
String path = "image/"+num+".png";
ImageIcon icon = MapTest.createImageIcon(path);
g.drawImage(icon.getImage(), 35*j, 15*i, 35, 15, null);
}
}
}
//重绘挡板
Fender f=new Fender(this);

f.createFender(g,Fender.x,630,100,20);
//重绘小球
Ball ball=new Ball();
ball.createBall(g, Ball.x0 , Ball.y0);


}

}

到这里基本的游戏已成型了,但是发现砖块消掉不完全或则还没被碰到的砖块已经被擦掉了,所以要重新启动一个线程来不停的对画图区域进行刷新,具体代码如下所示:
//刷新画布监听线程
class PaintThread extends Thread{

public void run(){
while(!Ball.isStop){
while(!Ball.isPause){
//重绘
repaint();


try{
Thread.sleep(1);
}catch(Exception ef){
ef.printStackTrace();
}

}

try{
Thread.sleep(1);
}catch(Exception ef){
ef.printStackTrace();
}

}
}

}

这样弄之后把上面的问题解决了,但是又发现挡板 、小球在不停的闪动,这时就需要根据双缓冲原理在swing中实现消除闪烁,所以上面重绘的代码改动如下:
//重绘
class mypanel extends JPanel{

public void paint(Graphics g){

//重写双缓冲机制
offSreenImage = this.createImage(495, 650);

//获得截取图片的画布
Graphics gImage = offSreenImage.getGraphics();

//获取画布的底色并且使用这种颜色填充画布,如果没有填充效果的话,则会出现拖动的效果
gImage.setColor(gImage.getColor());

//有清楚上一步图像的功能,相当于gImage.clearRect(0, 0, WIDTH, HEIGHT)  
gImage.fillRect(0, 0, 495, 650);


//重写父类的方法
super.paint(gImage);
//遍历砖块数组,实现重绘
for(int i=0;i<arr.length;i++){
for(int j=0;j<arr[i].length;j++){
if(arr[i][j]!=0){
int num=arr[i][j];
String path = "image/"+num+".png";
ImageIcon icon = MapTest.createImageIcon(path);
gImage.drawImage(icon.getImage(), 35*j, 15*i, 35, 15, null);
}
}
}
//重绘挡板
Fender f=new Fender(this);

f.createFender(gImage,Fender.x,630,100,20);
//重绘小球
Ball ball=new Ball();
ball.createBall(gImage, Ball.x0 , Ball.y0);


// 将接下来的图片加载到窗体画布上去,才能考到每次画的效果  
        g.drawImage(offSreenImage, 0, 0, null);

}

}
到此,一个打砖块的游戏已基本实现,但是这个游戏还很简单,还有很多问题,还需要很大得改进。但这里主要目的是深刻理解多线程以及在这个过程中的收获。

  • 大小: 21.9 KB
0
3
分享到:
评论

相关推荐

    qt打砖块游戏(原创)

    本项目以经典的打砖块游戏为蓝本,结合了Qt的功能,提供了丰富的视觉效果和交互体验。 在【Qt打砖块游戏】中,主要涉及以下几个关键知识点: 1. **Qt框架**:Qt库提供了大量的类和函数,用于构建窗口、控件、布局...

    基于JAVA的打砖块游戏

    通过学习和分析这个基于Java的打砖块游戏项目,初学者不仅能巩固Java编程基础,还能深入理解游戏开发的基本流程和技术,为今后的项目开发积累宝贵经验。在实际操作中,可以尝试修改代码以增加新的功能,如添加更多的...

    打砖块小游戏带音效

    打砖块游戏起源于1976年的街机游戏《Breakout》,玩家通过控制一块可移动的挡板来反弹一个小球,使小球击碎屏幕上排列的砖块。每击碎一块砖块,玩家得分,全部砖块消除后进入下一关。在此基础上,《打砖块小游戏带...

    Java打砖块游戏完整源码直接运行

    Java打砖块游戏是一款经典的计算机游戏,通过编程实现,它能帮助开发者理解基本的游戏逻辑、图形用户界面(GUI)设计以及事件处理等编程概念。在这个项目中,源码是使用Java编程语言在Eclipse集成开发环境中编写的。...

    Qt打砖块游戏

    【Qt打砖块游戏】是一款基于Qt框架开发的原创小游戏,它融合了经典的打砖块玩法与现代游戏设计元素。...通过分析和修改源代码,我们可以深入理解Qt的事件处理、图形渲染和多线程等核心概念,提升自己的编程技能。

    ios-打砖块.zip

    《iOS打砖块游戏源码解析》 ...通过分析这个打砖块游戏的源码,开发者不仅可以学习到iOS游戏开发的基本流程,还能深入理解面向对象编程、图形渲染、物理模拟等多个技术领域,为今后的项目开发积累宝贵经验。

    打砖块vb.net

    在这个项目中,我们将使用VB.NET(Visual Basic .NET)来创建一个简单的打砖块游戏。VB.NET是微软开发的一种面向对象的编程语言,它基于.NET Framework,提供了丰富的库和工具,非常适合初学者和专业开发者进行...

    自己有JAVA语言编写的打砖块小游戏

    4. **多线程**:为了实现游戏的实时性,小球的移动和其他游戏逻辑可能在单独的线程中运行,这样可以保证游戏在等待用户输入时不会阻塞。 5. **面向对象编程**: - `ball.java`:表示小球类,可能包含了小球的位置...

    打砖块 C++

    3. **对象导向编程 (OOP)**:打砖块游戏中的各种元素,如球、挡板、砖块,都可以抽象成类。每个类都包含它们各自的属性(如位置、速度)和方法(如移动、碰撞检测)。通过 OOP,我们可以更好地组织和管理代码。 4. ...

    打砖块.zip欢迎大家下载哟~

    2. **图形用户界面(GUI)**:打砖块游戏需要一个交互界面,这通常需要利用C++库如Qt或SDL来创建。这些库提供了窗口管理、事件处理、图像绘制等功能,帮助开发者构建游戏画面。 3. **游戏循环**:游戏的核心是游戏...

    java打砖块

    通过"Java打砖块"这个游戏项目,开发者可以深入理解Java编程的基础知识,同时提升面向对象设计、图形界面编程、多线程处理和算法应用等方面的能力。无论是对于初学者还是有经验的开发者,这都是一个富有挑战性和乐趣...

    基于java的打砖块游戏毕业(设计)论文.doc

    本篇毕业设计论文的主题是“基于Java的打砖块游戏开发”,旨在通过这个游戏实例来学习和掌握Java编程基础以及小型Java游戏开发的技术。 一、选题目的与Java游戏现状 选择开发基于Java的打砖块游戏,主要是因为打...

    Delphi PlateBall 3d打砖块游戏源程序.zip

    总的来说,通过对"Delphi PlateBall 3D打砖块游戏源程序"的学习,我们可以了解到Delphi在图形界面设计、事件驱动编程、3D图形渲染以及文件操作等多个方面的应用,对于提升Delphi编程技能和理解游戏开发流程具有很高...

    用java写了一个简单的打砖块游戏的游戏逻辑.zip

    4. **多线程**:在打砖块游戏中,球的运动与用户对挡板的操作通常是并行进行的,这需要使用到Java的多线程。可能有一个线程处理游戏逻辑(球的运动、碰撞检测),另一个线程处理用户输入。 5. **碰撞检测**:游戏的...

    同时玩30个球的打砖块游戏演示

    总结来说,"同时玩30个球的打砖块游戏"利用了特定的碰撞检测策略、并发处理、物理模拟以及资源管理等技术,实现了高效且富有挑战性的游戏体验。这样的设计不仅考验了开发者的编程技巧,也为玩家提供了独特的游戏乐趣...

    基于Java Swing的打砖块游戏.zip

    【标题】: "基于Java Swing的打砖块游戏" 是一个使用Java编程语言中的Swing库来实现的经典娱乐游戏。Swing是Java Standard Edition (Java SE)的一部分,它提供了丰富的图形用户界面(GUI)组件,使得开发者能够创建...

    JAVA编程小游戏打砖块

    9. **多线程**:为了实现流畅的游戏体验,游戏循环通常在单独的线程上运行,与UI线程分离,以防止阻塞。 10. **资源管理**:游戏可能包含音效和图像资源,需要适当地加载和释放,避免内存泄漏。 在这个"JAVA编程小...

    BrickGame一个基于Java Swing的打砖块游戏.zip

    《基于Java Swing的打砖块游戏——BrickGame解析》 BrickGame是一个采用Java Swing技术开发的简单但趣味盎然的打砖块游戏。它不仅是一个娱乐项目,更是Java编程爱好者进行毕业设计、课程设计或提升编程技能的理想...

    java- 打砖块小游戏

    在Java编程语言中,开发一个打砖块小游戏是一项常见的练习,它可以帮助初学者熟悉面向对象编程、游戏逻辑设计以及图形用户界面(GUI)的构建。这个项目通常涉及到以下几个关键知识点: 1. **Java基础**:首先,你...

Global site tag (gtag.js) - Google Analytics