美文网首页
并发编程-notify和wait

并发编程-notify和wait

作者: zhaoyang66 | 来源:发表于2020-01-30 15:28 被阅读0次

    通过wait和notify来实现发布订阅。就是线程1(生产者)生产一个数字,线程2(消费者)消费一个数字。消费的速度决定于生产者的速度。
    主对象 Mymain.java

    package tv.lab.thread;
    
    public class Mymain {
        public static void main(String[] args) {
            Ticket ticket = new Ticket();
            //创建2个任务,分别是r01和r02,并分别用2个线程运行这2个任务。注意,这里是2个线程,每个线程上有一个独立的任务。线程共享的对象是ticket
            Runnable01 r01 = new Runnable01(ticket);  
            Runnable02 r02 = new Runnable02(ticket);
            Thread t1 = new Thread(r01, "t1");  //声明线程t1
            Thread t2 = new Thread(r02, "t2");  //声明线程t2
    
            t1.start();
            t2.start();
        }
    }
    

    共享对象Ticket.java

    package tv.lab.thread;
    
    public class Ticket {
        int num = 0;
        boolean hasTicket = false;
    }
    

    生产对象Runnable01.java,这个是要生产数字的,产出的数字供消费者任务Runnable02.java消费

    package tv.lab.thread;
    
    import java.util.concurrent.TimeUnit;
    
    public class Runnable01 implements Runnable {
    
        Ticket ticket;
    
        public Runnable01(Ticket ticket) {
            this.ticket = ticket;  //将共享对象通过构造函数传递过来
        }
    
        @Override
        public void run() {
            while (true) {
                try {
                    TimeUnit.MILLISECONDS.sleep(10);  //休眠10毫秒
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
    
                synchronized (Ticket.class) { 
                    if (ticket.hasTicket == true) {
                        try {
                            Ticket.class.wait();  //如果为true,就是有票情况下,就等待,直到票被消费掉后,再产生新的票,如果不被消费,就一直挂起。
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    } else {
                        System.out.println(Thread.currentThread().getName() + " " + ++ticket.num);
                        ticket.hasTicket = true;
                        Ticket.class.notify();  //将等待的线程激活
                    }
                }
            }
        }
    }
    

    对上面代码的说明

    • 将Runnable01.java中的Ticket.class 替换为 ticket对象可以不可以?当然可以,只有是线程公用的对象即可,无论是Ticket.class或ticket,都是线程之间公用的对象。只有公用的对象,才可以用作锁。
      思考:将Ticket.class换为new Ticket()可以吗?肯定不可以,因为2个线程有2个不同的线程任务(Runnable01.java Runnable02.java),如果用new Ticket()就创建了2个不同的对象,不同对象是不能用在synchronized()同步锁中的,同步锁必须是线程之间公用的对象。
    • 如果将synchronized (Ticket.class) 这个同步代码块去掉可以吗?不可以,wait()和notify()必须在synochronized()同步代码块儿内才行。 意思就是说wait和notify是在获取对象锁的情况下才能使用,没有获取对象锁的情况下,是不能使用wait和notify的。

    消费任务Runnable02.java,当生产者生产后,这个任务用来消费任务

    package tv.lab.thread;
    
    public class Runnable02 implements Runnable {
    
        Ticket ticket;
    
        public Runnable02(Ticket ticket) {
            this.ticket = ticket;
        }
    
        @Override
        public void run() {
            while (true) {
    
                synchronized (Ticket.class) {
                    if (ticket.hasTicket == false) {
                        try {
                            Ticket.class.wait();
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    } else {  //如果有票,就打印出来
                        System.out.println("   " + Thread.currentThread().getName() + " " + ticket.num);
                        ticket.hasTicket = false;
                        Ticket.class.notify();
                    }
                }
            }
        }
    }
    

    执行结果如下:
    生产和消费成对出现。当生产一个数字后,就等待(wait)消费任务来消费,并通知消费任务来消费。当消费任务消费后,就挂起,通知生产任务生产。


    1580369096121.jpg

    相关文章

      网友评论

          本文标题:并发编程-notify和wait

          本文链接:https://www.haomeiwen.com/subject/cfbithtx.html