Java编写掷骰子游戏

废话不多说了,直接奔主题。

**多线程&&观察者模式

题目要求:《掷骰子》窗体小游戏,在该游戏中,玩家初始拥有1000的金钱,每次输入押大还是押小,以及下注金额,随机3个骰子的点数,如果3个骰子的总点数小于等于9,则开小,否则开大,然后判断玩家是否押对,如果未押对则扣除下注金额,如果押对则奖励和玩家下注金额相同的金钱。

分析:这个题目要求灵活运用多线程的相关知识,达到点击开始按钮时,有3个线程启动,分别控制3颗骰子的转动,在3颗骰子全部转完以后,回到主线程计算游戏结果。

//个线程控制颗骰子

Thread t = new Thread();

Thread t = new Thread();

Thread t = new Thread();

//启动个线程

t.start();

t.start();

t.start();

//将个线程加入主线程

t.join();

t.join();

t.join();

But,,,写完代码以后发现,这样做虽然能够保证游戏能够正确运行,但是当我点击开始按钮时,由于3个骰子线程都是直接开在主线程上的,点击开始按钮时,按钮出现下沉情况,子线程一直在后台运行,我窗体中的图片根本不会发生改变,而是直接显示最后的结果,意思就是骰子一直在后台转动,不在前台的窗体中及时更新显示。后来在网上苦苦找寻,大神们说如果想要通过点击JButton使窗体中的JLabel/JTextFeild等其他组件及时更新,直接在JButton的监听事件的实现方法里面直接创建匿名线程,也就是说直接在actionPerformed()方法中修改代码即可,这样能保证你的组件中内容的及时变换,实现非常炫酷的效果。

代码如下:

public void actionPerformed(ActionEvent e) {

new Thread(new Runnable() {

@Override

public void run() {

//将外部线程类转移到窗体内部

}

}).start();

}

 But,,,But,,,   虽然非常炫酷了,能够实现图片的及时更新了,游戏结果却错了,每次我的骰子还在转动呢,我的游戏结果却早早的就出来了。

原因:3根骰子线程属于子线程,窗体线程属于主线程,问题就在于:子线程可以通过变成精灵线程来保持与主线程的同生死,但是主线程却无法控制子线程何时死亡,只有等待子线程执行完所属的run()方法,结束线程后才知道。

解决方法:在主线程(main)中开3个子线程(t1,t2,t3),在每个子线程上再开一个子子线程(t11,t21,t31)。

t1,t2,t3只运行一次,负责创建子子线程;t11,t21,t31每个线程运行多次,负责控制窗体中的图标及时更新。

这样主线程就不受子线程的影响,开始按钮也不回出现下沉的情况。

但是同样在此处使用join方法也是hold不住子线程的,毕竟t1,t2,t3只运行了一次,join对他们来说根本不起作用,想要掌控t11,t21,t31,最容易理解的办法,就是使用观察者模式了。

将窗体看做观察者,子线程看做被观察者。子线程运行完时,通知观察者我已经运行完成,当观察者观察到子线程全都运行完时,才开始运行后续步骤。

全部代码:

1.窗体

package com.sxt.dice;

import java.awt.Color;

public class DiceFrame extends JFrame implements ActionListener, Observer {

/**

* 《掷骰子》控制台小游戏,在该游戏中,玩家初始拥有的金钱,每次输入押大还是押小,

* 以及下注金额,随机个骰子的点数,如果个骰子的总点数小于等于,则开小,否则开大,

* 然后判断玩家是否押对,如果未押对则扣除下注金额,如果押对则奖励和玩家下注金额相同的金钱。

*

* 运用观察者模式 个子线程分别控制个骰子,都已经结束时,通知观察者窗体,窗体观察到所有子线程都结束时,计算游戏结果

*

*/

private static final long serialVersionUID = L;

private JTextField txtPut;

private JButton btnStart;

private JLabel labResult;

private JComboBox<String> comboBox;

private JLabel labBigOrSmall;

private JLabel labPut;

private JLabel labSumMoney;

private JLabel labDice;

private JLabel labDice;

private JLabel labDice;

private JLabel labSum;

private JLabel labMes;

private static List<Icon> imgs = new ArrayList<Icon>();

public static void main(String[] args) {

new DiceFrame();

}

public DiceFrame() {

this.setLocationRelativeTo(null);

this.setBounds(, , , );

this.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);

getContentPane().setLayout(null);

this.setResizable(false);

labDice = new JLabel("");

labDice.setIcon(new ImageIcon("img/dices.jpg"));

labDice.setBounds(, , , );

getContentPane().add(labDice);

labSum = new JLabel("\u\uF\uD\uD\uFFA");

labSum.setBounds(, , , );

getContentPane().add(labSum);

labDice = new JLabel("");

labDice.setIcon(new ImageIcon("img/dices.jpg"));

labDice.setBounds(, , , );

getContentPane().add(labDice);

labDice = new JLabel("");

labDice.setIcon(new ImageIcon("img/dices.jpg"));

labDice.setBounds(, , , );

getContentPane().add(labDice);

labSumMoney = new JLabel("");

labSumMoney.setForeground(Color.red);

labSumMoney.setBounds(, , , );

getContentPane().add(labSumMoney);

labPut = new JLabel("\uC\uB\uEB\uCE\uFFA");

labPut.setToolTipText(".");

labPut.setBounds(, , , );

getContentPane().add(labPut);

txtPut = new JTextField();

txtPut.setBounds(, , , );

getContentPane().add(txtPut);

txtPut.setColumns();

labBigOrSmall = new JLabel("\uBC\uFFA");

labBigOrSmall.setBounds(, , , );

getContentPane().add(labBigOrSmall);

comboBox = new JComboBox<String>();

comboBox.setBounds(, , , );

getContentPane().add(comboBox);

comboBox.addItem("大");

comboBox.addItem("小");

labResult = new JLabel("");

labResult.setBounds(, , , );

getContentPane().add(labResult);

btnStart = new JButton("START");

btnStart.setBounds(, , , );

getContentPane().add(btnStart);

labMes = new JLabel("<html><font size= color=red>*</font></html>");

labMes.setBounds(, , , );

getContentPane().add(labMes);

this.setVisible(true);

imgs.add(new ImageIcon("img/.png"));

imgs.add(new ImageIcon("img/.png"));

imgs.add(new ImageIcon("img/.png"));

imgs.add(new ImageIcon("img/.png"));

imgs.add(new ImageIcon("img/.png"));

imgs.add(new ImageIcon("img/.png"));

btnStart.addActionListener(this);

}

@Override

public void actionPerformed(ActionEvent e) {

if (e.getSource() == btnStart) {

// 清除上次游戏的结果

labResult.setText("");

// 获取当前下注金额,用户余额,用户押大还是押小

String txt = txtPut.getText().trim();

String remain = labSumMoney.getText().trim();

// 余额不足,不能开始游戏,提示用户充值

if (Integer.parseInt(remain) <= ) {

JOptionPane.showMessageDialog(null, "当前余额不足,请充值!");

return;

}

// 下注金额合法性检查

if (txt.length() == ) {

// 提示用户输入

labMes.setText("*请输入下注金额");

labMes.setForeground(Color.RED);

return;

}

// 检查用户下注金额是否在有效范围内

if (Integer.parseInt(txt) <=

|| Integer.parseInt(txt) > Integer.parseInt(remain)) {

txtPut.setText("");

labMes.setText("下注金额应在~" + remain + "之间");

return;

}

// 游戏开始后相关项不可更改

txtPut.setEnabled(false);

labMes.setText("");

comboBox.setEnabled(false);

//在主线程上开t,t,t 个子线程

Thread t = new Thread() {

@Override

public void run() {

//每个子线程上再开子子线程,控制图标变换

IconThread t = new IconThread(labDice, imgs);

//给t添加观察者,即当前窗体

t.addObserver(DiceFrame.this);

new Thread(t).start();

}

};

Thread t = new Thread() {

@Override

public void run() {

IconThread t = new IconThread(labDice, imgs);

t.addObserver(DiceFrame.this);

new Thread(t).start();

}

};

Thread t = new Thread() {

@Override

public void run() {

IconThread t = new IconThread(labDice, imgs);

t.addObserver(DiceFrame.this);

new Thread(t).start();

}

};

t.start();

t.start();

t.start();

}

}

/**

* 获取骰子点数和

*

* @param lab

* @return sum

*/

private int result(JLabel lab) {

// 获取当前骰子图片

Icon icon = lab.getIcon();

int sum = ;

for (int i = ; i < imgs.size(); i++) {

if (icon.equals(imgs.get(i))) {

sum += (i + );

break;

}

}

return sum;

}

// 构建所有被观察者的集合

Vector<Observable> allObservables = new Vector<Observable>();

@Override

public void update(Observable o, Object arg) {

System.out.println(o + ".................");

// 如果集合中不包含当前被观察者,将此被观察者加入集合

if (allObservables.contains(o) == false) {

allObservables.add(o);

}

// 如果集合中被观察者个数为,说明个骰子线程已经全部结束

if (allObservables.size() == ) {

// 获取当前下注金额,用户余额,用户押大还是押小

String txt = txtPut.getText().trim();

String remain = labSumMoney.getText().trim();

String bigOrSmall = comboBox.getSelectedItem().toString();

// 获取每个骰子点数

int sum = result(labDice);

int sum = result(labDice);

int sum = result(labDice);

System.out.println(sum + "-" + sum + "-" + sum);

int sum = sum + sum + sum;

System.out.println(sum);

if (sum > && "大".equals(bigOrSmall) || sum <=

&& "小".equals(bigOrSmall)) {

// 奖励玩家相应金额

remain = String.valueOf(Integer.parseInt(remain)

+ Integer.parseInt(txt));

labSumMoney.setText(remain);

// 显示游戏结果

labResult.setText("WIN");

labResult.setForeground(Color.GREEN);

labResult.setFont(new Font("宋体", Font.BOLD, ));

} else {

// 扣除玩家相应金额

remain = String.valueOf(Integer.parseInt(remain)

- Integer.parseInt(txt));

labSumMoney.setText(remain);

labResult.setText("FAIL");

labResult.setForeground(Color.red);

labResult.setFont(new Font("宋体", Font.BOLD, ));

}

txtPut.setEnabled(true);

comboBox.setEnabled(true);

// 本次游戏结束后移除集合中所有线程

allObservables.removeAll(allObservables);

}

}

}

2.线程

package com.sxt.dice;

import java.util.List;

import java.util.Observable;

import java.util.Random;

import javax.swing.Icon;

import javax.swing.JLabel;

public class IconThread extends Observable implements Runnable {

/**

* 运用观察者模式,将子线程作为被观察对象,一旦子线程运行完,发生改变,通知观察者

*/

JLabel lab;

Random random = new Random();

List<Icon> imgs;

public IconThread(JLabel lab, List<Icon> imgs) {

this.lab = lab;

this.imgs = imgs;

}

@Override

public void run() {

//设置每颗骰子转动次

int count = ;

while (count > ) {

//获取一个随机数[~)

int index = random.nextInt();

//从imgs集合中取相应图片放入lab中

lab.setIcon(imgs.get(index));

count--;

try {

Thread.sleep();

} catch (InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

this.setChanged();// 子线程运行完,发生改变

this.notifyObservers();// 通知观察者

}

}

以上所述就是关于Java编写掷骰子游戏的全部内容,希望大家喜欢。

以上是 Java编写掷骰子游戏 的全部内容, 来源链接: utcz.com/p/208536.html

回到顶部