美文网首页
只唤醒对方线程,不唤醒本方线程(jdk5.0新特性)

只唤醒对方线程,不唤醒本方线程(jdk5.0新特性)

作者: 东风谷123Liter | 来源:发表于2018-07-12 09:37 被阅读0次

    Lock

    • 可视化锁,隐式锁边显示锁。

      • Lock( )获取锁

      • unLock( )释放锁

    • 支持多个Condition相关对象。

    • Lock接口:

      • Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。此实现允许更灵活的结构,可以具有差别很大的属性,可以支持多个相关的 Condition 对象。
    • Condition接口:

      • Condition 将 Object 监视器方法(waitnotifynotifyAll)分解成截然不同的对象,以便通过将这些对象与任意 Lock 实现组合使用,为每个对象提供多个等待 set(wait-set)。其中,Lock 替代了 synchronized 方法和语句的使用,Condition 替代了 Object 监视器方法的使用。
    • image.png
    • 总结

      • JDK5.0中提供了多线程升级解决方案。

      • 将同步synchronized替换成显示的Lcok操作。

      • 将Object中的wait, notify, notifyAll,替换成Condition对象。

      • 该对象可以通过Lock 锁进行获取。

      • 实现了本方只唤醒对方线程的操作!

    • 具体代码:

    /*
    生产者消费者问题。
    需求:有多个生产者生产商品,放到仓库中;多个消费者消费商品,消费者从仓库中取出上品进行消费;仓库里最多只能存放一个商品。
         。。。。
         用java5.0的 java.util.concurrent.locks 新特性实现线程之间的通信。
    分析:两个生产者线程,两个消费者线程;通过,创建继承Runnable接口的生产、消费类,再将相应的类型对象传递给Thread类来创建线程。
         调用java.util.concurrent.locks包中的Lock接口中的ReentrantLock类来创建锁。
         调用java.util.concurrent.locks包中的Condition接口中的lock.newCondition()来创建状态对象。
    */
    import java.util.concurrent.locks.*;    //调用java5.0的新特性,摒弃synchronized那啥啥的
    class Resource{     //资源仓库
        private int count = 0;      //商品数量
        private String name ;       //商品名称
        private boolean flag = false;   //用来标记仓库是否有商品
    
        private Lock lock = new ReentrantLock();    //创建一个锁对象
        private Condition condition_pro = lock.newCondition();  //创建生产者线程状态对象,并绑定到锁上。
        private Condition condition_con = lock.newCondition();  //创建消费者线程状态对象,并绑定到锁上。
    
        public void set(String name) throws InterruptedException{   //为了提高代码可读性,把创建商品的过程封装成方法,放到Resource类下面。
                lock.lock();    //打开锁,保证以下代码块同步
                try{    //检测异常
                    while(flag)     //让唤醒的线程循环访问flag标签;避免用if时,被唤醒的线程不在判断仓库中是否有商品而生产多个商品
                        condition_pro.await();
                    this.name = name+"---"+count++;     //生产商品
                    System.out.println(Thread.currentThread().getName()+"--生产者--"+this.name);   //生产商品
                flag = true;    //更改标签
                condition_con.signal();     //唤醒消费者线程
                }
                finally{    //一定会执行的代码:关闭锁!
                    lock.unlock();
                }
                
        }
        public  void out() throws InterruptedException{ //为了提高代码可读性,把消费商品的过程封装成方法,放到Resource类下面。
            lock.lock();
            try{
                while(!flag)    
                    condition_con.await();
                System.out.println(Thread.currentThread().getName()+"消费者"+this.name);
                flag = false;
                condition_pro.signal();     //唤醒生产者线程
            }
            finally{
                lock.unlock();
            }
        }
    }
    
    class Producer implements Runnable{     //生产类
        private Resource r;     //为了保证所有的线程操作的事同一个仓库对象,这里就没有再创建对象,只是创建引用。
        Producer(Resource r){   //引用对象lalsl
            this.r = r;
        }
        public void run(){      //线程的run()方法
            while(true){
                try{        //检测异常
                    r.set("商品");
                }
                catch(Exception e){}
            }
        }
    }
    
    class Consumer implements Runnable{     //消费类
        private Resource r; 
        Consumer(Resource r){
            this.r = r;
        }
        public void run(){
            while(true){
                try{
                    r.out();
                }
                catch(Exception e){}
            }
        }
    }
    
    class LockDemo{
        public static void main(String[] args){
            Resource r = new Resource();    //创建仓库
    
            Producer p = new Producer(r);
            Consumer c = new Consumer(r);
    
            Thread t1 = new Thread(p);  //创建多个生产者,多个消费者;
            Thread t2 = new Thread(p);
            Thread t3 = new Thread(c);
            Thread t4 = new Thread(c);
    
            t1.start();
            t2.start();
            t3.start();
            t4.start();
        }
    }
    

    相关文章

      网友评论

          本文标题:只唤醒对方线程,不唤醒本方线程(jdk5.0新特性)

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