美文网首页
只唤醒对方线程,不唤醒本方线程(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