双线程猜数字 java多线程习题详述


文章目录

  • 双线程猜数字 java多线程习题详述
  • 1、实验要求:
  • 2、代码
  • 3、遇到的问题及解决办法
  • 3.1、如何生成1~100的随机整数
  • Java random() 方法
  • 3.2、如何让两个线程互相联系
  • 线程通信
  • wait/notify 机制
  • 4、总结(报告中的实验内容)
  • 5、想要更深入的了解多线程


1、实验要求:

用两个线程玩猜数字游戏,第一个线程负责随机给出1到100之间的一个整数,第二个线程负责猜出这个数。要求每当第二个线程给出自己的猜测后,第一个线程都会提示“猜小了”、“猜大了”或“猜对了”。猜数之前,要求第二个线程要等待第一个线程设置好要猜测的数。第一个线程设置好猜测数之后,两个线程还要互相等待,其原则是,第二个线程给出自己的猜测后,等待第一个线程给出的提示;第一个线程给出提示后,等待给第二个线程给出猜测,如此进行,直到第二个线程给出正确的猜测后,两个线程进入死亡状态。

2、代码

package Thread1;

public class TwoThreadGuessNumber {
    public static void main(String[] args) {

        Number number=new Number();
        number.giveNumberThread.start();
        number.guessNumberThread.start();

    }
}
package Thread1;

public class Number implements Runnable{
    final int SMALLER=-1,LARGER=1,SUCCESS=8;
    int realNumber,guessNumber,min=0,max=100,message=SMALLER;
    boolean pleaseGuess=false,isGiveNumber=false;
    Thread giveNumberThread,guessNumberThread;
    Number() {
          giveNumberThread= new Thread(this);
          guessNumberThread= new Thread(this);
 //【代码1】创建giveNumberThread,当前Number类的实例是giveNumberThread的目标对象
 //【代码2】创建guessNumberThread,当前Number类的实例是guessNumberThread的目标对象
    }
    public void run() {
        for(int count=1;true;count++) {
            setMessage(count);
            if( message==SUCCESS)
                return;
        }
    }
    public synchronized void setMessage(int count) {
        if(Thread.currentThread()==giveNumberThread&&isGiveNumber==false) {
            realNumber=(int)(Math.random()*100)+1;
            System.out.println("随机给你一个1至100之间的数,猜猜是多少?");
            isGiveNumber=true;
            pleaseGuess=true;
        }
        if(Thread.currentThread()==giveNumberThread) {
            while(pleaseGuess==true)
                try  { wait();  //让出CPU使用权,让另一个线程开始猜数
                }
                catch(InterruptedException e){}
            if(realNumber>guessNumber)  { //结束等待后,根据另一个线程的猜测给出提示
                message=SMALLER;
                System.out.println("你猜小了");
            }
            else if(realNumber<guessNumber) {
                message=LARGER;
                System.out.println("你猜大了");
            }
            else {
                message=SUCCESS;
                System.out.println("恭喜,你猜对了");
            }
            pleaseGuess=true;
        }

        if(Thread.currentThread()==guessNumberThread&&isGiveNumber==true) {
            while(pleaseGuess==false)
                try { wait();  //让出CPU使用权,让另一个线程给出提示
                }
                catch(InterruptedException e){}
            if(message==SMALLER) {
                min=guessNumber;
                guessNumber=(min+max)/2;
                System.out.println("我第"+count+"次猜这个数是:"+guessNumber);
            }
            else if(message==LARGER) {
                max=guessNumber;
                guessNumber=(min+max)/2;
                System.out.println("我第"+count+"次猜这个数是:"+guessNumber);
            }
            pleaseGuess=false;
        }
        notifyAll();
    }
}
运行结果:
随机给你一个1至100之间的数,猜猜是多少?
我第1次猜这个数是:50
你猜小了
我第2次猜这个数是:75
你猜大了
我第3次猜这个数是:62
你猜小了
我第4次猜这个数是:68
恭喜,你猜对了

进程已结束,退出代码 0

3、遇到的问题及解决办法

3.1、如何生成1~100的随机整数

Java random() 方法

random()方法用于返回一个随机数,随机数范围为 0.0 =< Math.random < 1.0

语法:

static double random()

参数:

这是一个默认方法,不接受任何参数。

返回值

该方法返回 double 值。

实例:

public class Test{
    public static void main(String args[]){
        System.out.println( Math.random() );
        System.out.println( Math.random() );
    }
}

3.2、如何让两个线程互相联系

线程通信

为了实现线程一与线程二之间的联系,采用线程通信的方式。(wait和notify)

概述:

线程与线程之间不是相互独立的个体,它们彼此之间需要相互通信和协作,虽然两个线程实现了通信,但是凭借线程B不断地通过while语句轮询来检测某一个条件,这样会导致CPU的浪费。因此,需要一种机制来减少CPU资源的浪费,而且还能实现多个线程之间的通信,即 wait/notify 机制。

wait/notify 机制

1.wait()——让当前线程 (Thread.concurrentThread() 方法所返回的线程) 释放对象锁并进入等待(阻塞)状态。

2.notify()——唤醒一个正在等待相应对象锁的线程,使其进入就绪队列,以便在当前线程释放锁后竞争锁,进而得到CPU的执行。

3.notifyAll()——唤醒所有正在等待相应对象锁的线程,使它们进入就绪队列,以便在当前线程释放锁后竞争锁,进而得到CPU的执行。

4、总结(报告中的实验内容)

使用Thread创建线程对象时,通常使用的构造方法是:

Thread(Runnable target);

该构造方法中的参数是一个Runnable类型的接口,因此,在创建线程对象时必须向构造方法的参数传递一个实现Runnable接口类的实例,该实例对象称作所创线程的目标对象。当线程调用start方法后,一旦轮到它来享用CPU资源,目标对象就会自动调用接口中的run方法(接口回调),这一过程是自动实现的,用户程序只需要让线程调用start方法即可。线程绑定于Runnable接口,也就是说,当线程被调度并转入运行状态时,所执行的就是run()方法中所规定的操作。

线程同步是指几个线程都需要调用同一个同步方法(用synchronized修饰的方法)。一个线程在使用的同步方法中时,可能根据问题的需要,必须使用wait()方法暂时让出CPU的使用权,以便其它线程使用这个同步方法。其它线程如果在使用这个同步方法时如果不需要等待,那么它用完这个同步方法的同时,应当执行notifyAll()方法通知所有的由于使用这个同步方法而处于等待的线程结束等待。曾中断的线程就会从刚才的中断处继续执行这个同步方法,并遵循“先中断先继续”的原则。如果使用notify()方法,那么只是通知处于等待中的线程的某一个结束等待。wait()、notify()和notifyAll()都是Object类中的final方法,被所有的类继承、且不允许重写的方法

5、想要更深入的了解多线程

如果小伙伴想要更加深入的了解多线程,点击此黄色字体链接,java多线程详解教程,相信您会有更大的收获。