转载请标明博客的地址

本人博客和github账号,如果对你有帮助请在本人github项目AioSocket上点个star,激励作者对社区贡献

虽然代码有点多,但是这样设计也是为了程序的可扩展性,和解耦合

比赛规则如下:

1.有四个选手, A1和A2为一个队, B1和B2为另一个队. A1首先发球(启动球), 然后B1, A2, B2将最后发球. 每一轮每个选手发6个球.

2.选手不改变他们的位置.

3.比赛期间, 双方选手必须轮流发球,并且在同一个队伍的两个选手可以竞争发球.

4.当轮到某个选手时, 他/她可以调用一个叫做shot(rate) 的随机函数来模拟比赛,在给定概率rate以内,该函数返回 “in”, 否则返回”out”. 例如 rate=85%, 则球在界内的概率为85%, 出界的概率为15%.

5.如果shot函数返回”in”, 对方选手必须调用shot函数把球打回.

6.如果shot函数返回”out”, 对方选手赢得1分,随后重新发球.

7.当每个选手发完6个球后比赛终止.分数多的一方赢得比赛.分数一样多,比赛为平局.

8.每个选手作为一个线程实现.

直接上源程序:

package Test;

import java.util.HashMap;
import java.util.concurrent.CountDownLatch;
import java.util.logging.Level;
import java.util.logging.Logger;
/**

* @author haibiscuit(本人博客名)
*/
public class TableTennisThreadDemo {

public static void main(String args[]) {

CountDownLatch countDownLatch = new CountDownLatch(4);
double rate = 0.75; //设置选手赢的概率
int sendNum = 6; //每个选手发球次数

//两个队的名字
String team1_Name = TeamController.getTeamName()[0];
String team2_Name = TeamController.getTeamName()[1];

//四个选手名字
String team1_memName1 = TeamController.getTeamMems(team1_Name)[0];
String team1_memName2 = TeamController.getTeamMems(team1_Name)[1];
String team2_memName1 = TeamController.getTeamMems(team2_Name)[0];
String team2_memName2 = TeamController.getTeamMems(team2_Name)[1];

//创建两个队
Team team1 = new Team(team1_Name);
Team team2 = new Team(team2_Name);

//设置对手
TeamController.setTeamRival(team2, team1);
//设置比赛场数
TeamController.setMatchTimes(sendNum*4); //参数为选手数*每个选手的发球数

//创建乒乓球对象
TableTennis tableTennis = new TableTennis(rate);

//创建四个线程类
Thread thread1 = new Thread(new TeamMember(team1_memName1, sendNum, team1, tableTennis, countDownLatch));
Thread thread2 = new Thread(new TeamMember(team1_memName2, sendNum, team1, tableTennis, countDownLatch));
Thread thread3 = new Thread(new TeamMember(team2_memName1, sendNum, team2, tableTennis, countDownLatch));
Thread thread4 = new Thread(new TeamMember(team2_memName2, sendNum, team2, tableTennis, countDownLatch));
try {
Thread.sleep(1000);
} catch (InterruptedException ex) {
Logger.getLogger(TableTennisThreadDemo.class.getName()).log(Level.SEVERE, null, ex);
}

thread1.start();
thread2.start();
thread3.start();
thread4.start();

//等待四个选手运行完
try {
countDownLatch.await();
System.out.println("team1得分:"+team1.getScore());
System.out.println("team2得分:"+team2.getScore());
if(team1.getScore()>team2.getScore()){
System.out.println("恭喜team1队赢!");
}else if(team1.getScore()<team2.getScore()){
System.out.println("恭喜team2队赢!");
}else{
System.out.println("平局了!");
}
} catch (InterruptedException ex) {
Logger.getLogger(TableTennisThreadDemo.class.getName()).log(Level.SEVERE, null, ex);
}
}
}

class TableTennis {

int rate; //赢的频率
volatile int state = 0; //球的状态,0代表发球,1代表传球,默认是发球状态 
String trail = "team1"; //传球的轨迹
String teamTrail = "team1"; //发球的轨迹
volatile boolean isEnd = false; //判断比赛有没有结束
volatile int totalSendNum = 0; //总共发球次数

public int getTotalSendNum() {
return totalSendNum;
}

public void addTotalSendNum() {
this.totalSendNum = ++totalSendNum;
}

public boolean isIsEnd() {
return isEnd;
}

public void setIsEnd(boolean isEnd) {
this.isEnd = isEnd;
}

public String getTeamTrail() {
return teamTrail;
}

public void setTeamTrail(String teamTrail) {
this.teamTrail = teamTrail;
}

public TableTennis(double rate) {
this.rate = (int) (rate * 100);
}

/**
* 判断输赢
*
* @return
*/
public final String shot() {
int randRate = (int) (Math.random() * 100);
if (((100 - randRate) > (this.rate))) {
return "out";
} else {
return "in";
}
}

public int getRate() {
return rate;
}

public void setRate(int rate) {
this.rate = rate;
}

public int getState() {
return state;
}

public void setState(int state) {
this.state = state;
}

public String getTrail() {
return trail;
}

public void setTrail(String trail) {
this.trail = trail;
}

}

class Team {

String teamName; //球队的名称
int score = 0; //球队的得分

public Team(String teamName) {
this.teamName = teamName;
}

public String getTeamName() {
return teamName;
}

public void setTeamName(String teamName) {
this.teamName = teamName;
}

public int getScore() {
return score;
}

public void addScore() {
this.score = ++score;
}

}

class TeamController {

private static final String[] team = {"team1", "team2"}; //比赛队的名称
private static final HashMap<String, String[]> map = new HashMap(); //用于存放各队的队员信息
private static final HashMap<String, Team> rivalTeam = new HashMap(); //用于存放各队的对手
private static int totalMatchTime; //默认0场比赛

static {
map.put(team[0], new String[]{"A1", "A2"}); //初始化team1队球员 
map.put(team[1], new String[]{"B1", "B2"}); //初始化team2队球员
}

//设置各队的对方队伍
public static void setTeamRival(Team rivalTeam1, Team rivalTeam2) {
rivalTeam.put(team[0], rivalTeam1);
rivalTeam.put(team[1], rivalTeam2);
}
public static void setMatchTimes(int time){
totalMatchTime = time;
}
public static int getMatchTimes(){
return totalMatchTime;
}

public static Team getRivalTeam(String teamName) {
return rivalTeam.get(teamName);
}

public static String[] getTeamName() {
return team;
}

public static String[] getTeamMems(String teamName) {
return map.get(teamName);
}
}
//队员

class TeamMember implements Runnable {

Team team; //球员所在的球队
TableTennis tableTennis; //每个队员所传的球
String memName; //球员名称
int sendNum; //每个队员传球的次数

CountDownLatch countDownLatch; //保证每个球员发球次数

public TeamMember(String memName, int sendNum, Team team, TableTennis tableTennis, CountDownLatch countDownLatch) {
this.team = team;
this.tableTennis = tableTennis;
this.memName = memName;
this.sendNum = sendNum;
this.countDownLatch = countDownLatch;
}

@Override
public void run() {
while (!this.tableTennis.isIsEnd()) { //当在发球此数内
sendBall();
}
countDownLatch.countDown();
}

/**
* 发球的逻辑注意这里的synchronized,并发控制
*/
private void sendBall() {
synchronized (tableTennis) {

// System.out.println("当前发球线程:" + Thread.currentThread().getName());
while (!(this.tableTennis.getTrail().equals(this.team.getTeamName()))) { //判断是否是该队传球
// System.out.println("发球睡眠线程:" + Thread.currentThread().getName());
if(this.tableTennis.isIsEnd()){
return;
}

try {
tableTennis.wait();
} catch (InterruptedException ex) {
Logger.getLogger(TableTennis.class.getName()).log(Level.SEVERE, null, ex);
}
}
if(this.tableTennis.isIsEnd()){ //判断比赛有没有结束,结束直接返回
return;

if (this.tableTennis.getState() == 0) { //判断是不是发球状态
this.sendNum--; //将发球次数减一 
}

//发球的时候判断在界内还是界外
if (this.tableTennis.shot().equals("in")) { //发球表示球在界内
this.tableTennis.setTrail(TeamController.getRivalTeam(this.team.getTeamName()).getTeamName()); //对手接球
this.tableTennis.setState(1); //如果发球在界内,将发球状态改为传球状态
System.out.print(this.memName + "-in-");
} else { //如果在界外,此时还是发球状态
TeamController.getRivalTeam(this.team.getTeamName()).addScore(); //对方队加上一分
System.out.print(this.memName + "-out\n");
this.tableTennis.setTeamTrail(TeamController.getRivalTeam(this.tableTennis.getTeamTrail()).getTeamName());
this.tableTennis.setTrail(this.tableTennis.getTeamTrail());
this.tableTennis.setState(0);
this.tableTennis.addTotalSendNum(); //将发球次数加一
if(this.tableTennis.getTotalSendNum()==TeamController.getMatchTimes()){
this.tableTennis.setIsEnd(true);
}
}

//执行完唤醒对手线程
tableTennis.notifyAll();
}
}

}

实现思路:利用多线程的wait和notifyall的方法实现线程切换,核心类是TableTennis类

玩转Java多线程(乒乓球比赛)的更多相关文章

  1. 玩转java多线程(wait和notifyAll的正确使用姿势)

    转载请标明博客的地址 本人博客和github账号,如果对你有帮助请在本人github项目AioSocket上点个star,激励作者对社区贡献 个人博客:https://www.cnblogs.com/ ...

  2. 玩转Java多线程(Lock.Condition的正确使用姿势)

    转载请标明博客的地址 本人博客和github账号,如果对你有帮助请在本人github项目AioSocket上点个star,激励作者对社区贡献 个人博客:https://www.cnblogs.com/ ...

  3. Java多线程开发系列之四:玩转多线程(线程的控制2)

    在上节的线程控制(详情点击这里)中,我们讲解了线程的等待join().守护线程.本节我们将会把剩下的线程控制内容一并讲完,主要内容有线程的睡眠.让步.优先级.挂起和恢复.停止等. 废话不多说,我们直接 ...

  4. Java多线程可以分组,还能这样玩!

    前面的文章,栈长和大家分享过多线程创建的3种方式<实现 Java 多线程的 3 种方式>. 但如果线程很多的情况下,你知道如何对它们进行分组吗? 和 Dubbo 的服务分组一样,Java ...

  5. Java多线程之赛跑游戏

    在JavaSE中,多线程是一个重要的内容. 我们要了解多线程的概念,就要先了解进程的概念:要了解进程的概念,就离不开操作系统的概念. 在一台正常运行的电脑中,计算机硬件(如CPU.内存.硬盘.网卡.显 ...

  6. Java多线程之赛跑游戏(含生成exe文件)

    在JavaSE中,多线程是一个重要的内容. 我们要了解多线程的概念,就要先了解进程的概念:要了解进程的概念,就离不开操作系统的概念. 在一台正常运行的电脑中,计算机硬件(如CPU.内存.硬盘.网卡.显 ...

  7. Java最重要的21个技术点和知识点之JAVA多线程、时间处理、数据格式

    (四)Java最重要的21个技术点和知识点之JAVA多线程.时间处理.数据格式  写这篇文章的目的是想总结一下自己这么多年JAVA培训的一些心得体会,主要是和一些java基础知识点相关的,所以也希望能 ...

  8. 从JAVA多线程理解到集群分布式和网络设计的浅析

    对于JAVA多线程的应用非常广泛,现在的系统没有多线程几乎什么也做不了,很多时候我们在何种场合如何应用多线程成为一种首先需要选择的问题,另外关于java多线程的知识也是非常的多,本文中先介绍和说明一些 ...

  9. 关于java多线程理解到集群分布式和网络设计的浅析

    对于JAVA多线程的应用非常广泛,现在的系统没有多线程几乎什么也做不了,很多时候我们在何种场合如何应用多线程成为一种首先需要选择的问题, 另外关于java多线程的知识也是非常的多,本文中先介绍和说明一 ...

随机推荐

  1. s便携小方法,你值得拥有

    引言: 本章没有深奥的讲解js一些底层原理,比如this指针.作用域.原型啦,涉及的都是一些有利于平时开发时简化代码,提高执行效率,或者说可以当做一种经验方法来使用,篇幅都不长,小步快跑的让你阅读完整 ...

  2. numpy 辨异(三)—— hstack/column_stack,linalg.eig/linalg.eigh

    1. np.hstack np.column_stack >>> np.hstack([np.array([1, 2, 3]), np.array([4, 5, 6])]) arra ...

  3. [Scikit-Learn] - introduction

    scikit-learn是一个用于机器学习的 Python 模块,建立在SciPy基础之上. 主要特点: 操作简单.高效的数据挖掘和数据分析 无访问限制,在任何情况下可重新使用 建立在NumPy.Sc ...

  4. hdu 1087 Super Jumping! Jumping! Jumping!(dp 最长上升子序列和)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1087 ------------------------------------------------ ...

  5. 比较实时分布式搜索引擎(senseidb、Solr、elasticsearch)

    1.它们是基于lucene的. 2.它们分布:sensedb它是multi-write;Solr的shards它是master-slave状态.基于pull策略:elasticsearch的shard ...

  6. 參数传递(引用,指针,值传递)C++11

    C++中,函数的參数传递方式有值传递.地址传递.传地址有指针和引用方式. 在函数參数中,传地址的理由有: 1.使被调函数能够改动主调函数中的数据对象: 2.传地址能够降低数据拷贝,提高程序运行速度. ...

  7. 利用WPF的ListView进行大数据量异步加载

    原文:利用WPF的ListView进行大数据量异步加载 由于之前利用Winform的ListView进行大数据量加载的时候,诟病良多,所以今天试着用WPF的ListView来做了一下,结果没有让我失望 ...

  8. 跟我学ASP.NET MVC之十一:URL路由

    摘要: 在MVC框架之前,ASP.NET假定在请求的URLs和服务器硬盘文件之间有直接的关系.服务器的职责是接收浏览器请求,从相应的文件发送输出. 这种方法只能工作于Web表单,每一个ASPX页面既是 ...

  9. Arcgis for Javascript实现图

    首先,截个图给大家看结果: 初始化状态 放大后的状态 点击选中后的状态 如上图所看到的,一般的涉及到的地图的统计涉及到上述所展示的三个状态:1.初始化状态.2.缩放后的状态:3.点击选中显示详情状态. ...

  10. 文章之间的基本总结Activity生命周期

    子曰:溫故而知新,能够為師矣.<論語> 学习技术也一样,对于技术文档或者经典的技术书籍来说,指望看一遍就全然掌握,那基本不大可能,所以我们须要常常回过头再细致研读几遍,以领悟到作者的思想精 ...