Java多线程(5)-线程的通信

线程的通信

引入:
使用两个线程打印 1-100。线程1, 线程2 交替打印
这时如果只使用同步锁虽然可以交互打印,但无法完成交替打印

package JavaThread;
class Window implements Runnable{
    private int num=1;
   public void run(){
       while(true){
           synchronized(this){
               if(num<=100) {
                   try {
                       Thread.sleep(10);
                   } catch (InterruptedException e) {
                       e.printStackTrace();
                   }
                   System.out.println(Thread.currentThread().getName() + "打印:" + num);
                   num++;
               }else{
                   break;
               }
           }
       }
   }
}
public class Test {
    public static void main(String[] args) {
        Window w=new Window();
        Thread t1=new Thread(w);
        Thread t2=new Thread(w);

        t1.setName("线程一");
        t2.setName("线程二");

        t1.start();
        t2.start();
    }
}

如果要完成交替打印那么就要使用线程之间的通信来完成,当线程一打印完以后就要把cpu给线程二并且自己处于阻塞状态,线程二也是同理

需要使用的方法:
wait() 与 notify() 和 notifyAll(

  1. wait():令当前线程挂起并放弃CPU、同步资源并等待,使别的线程可访问并修改共享资源,而当前线程排队等候其他线程调用notify()或notifyAll()方法唤醒,唤醒后等待重新获得对监视器的所有
    权后才能继续执行。
  2. notify():唤醒正在排队等待同步资源的线程中优先级最高者结束等待
  3. notifyAll ():唤醒正在排队等待资源的所有线程结束等待. 

注意:

  • 这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报java.lang.IllegalMonitorStateException异常。
  • 因为这三个方法必须有锁对象调用,而任意对象都可以作为synchronized的同步锁,因此这三个方法只能在Object类中声明。

wait() 方法

  • 在当前线程中调用方法: 对象名.wait()
  • 使当前线程进入等待(某对象)状态 ,直到另一线程对该对象发出 notify
    (或notifyAll) 为止。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)
  • 调用此方法后,当前线程将释放对象监控权 ,然后进入等待
  • 在当前线程被notify后,要重新获得监控权,然后从断点处继续代码的执行。

notify()/notifyAll()

  • 在当前线程中调用方法: 对象名.notify()
  • 功能:唤醒等待该对象监控权的一个/所有线程。
  • 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)
class Number implements Runnable{
    private int number=1;
    public void run(){
        while(true){
            synchronized(this) {
                notify();
                if (number <= 100) {
                    try {
                        Thread.sleep(10);//进入阻塞不释放锁
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + ": " + number);
                    number++;
                    try {
                        //使得调用 wait()方法的线程进入阻塞状态会释放锁
                        wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    break;
                }
            }
        }
    }
}
public class Thread01 {
    public static void main(String[] args) {
        Number nu=new Number();
        Thread t1=new Thread(nu);
        Thread t2=new Thread(nu);
        t1.setName("线程1");
        t2.setName("线程2");
        t1.start();
        t2.start();
    }
}

当线程一刚进入同步代码块时便唤醒线程二,使线程二在外面等待,当线程一打印num后进入wait()状态,释放锁线程二便进入同步代码块,步骤同理。

sleep() 和 wait()的异同

  1. 相同点:一旦执行方法,都可以使得当前的线程进入阻塞状态。
  2. 不同点:1)两个方法声明的位置不同:Thread类中声明sleep() , Object类中声明wait()
  3. 调用的要求不同:sleep()可以在任何需要的场景下调用。 wait()必须使用在同步代码块或同步方法中
  4. 关于是否释放同步监视器:如果两个方法都使用在同步代码块或同步方法中,sleep()不会释放锁,wait()会释放锁。
发布了45 篇原创文章 · 获赞 43 · 访问量 7069

猜你喜欢

转载自blog.csdn.net/qq_42193790/article/details/104440990