以下讲解Lock线程同步通信,也是比synchronized强大的一个功能点

先看一个常规的案例:

用户类

public class Person {
    public void eat(){
        for(int i=0;i<3;i++){
            System.out.println("eat...");
        }
    }

    public void drink(){
        for(int i=0;i<3;i++){
            System.out.println("drink...");
        }
    }

    public void play(){
        for(int i=0;i<3;i++){
            System.out.println("play...");
        }
    }
}
public class LockConditionTest {
    public static void main(String[] args) {
        final Person person = new Person();
        new Thread(
                new Runnable() {
                    @Override
                    public void run() {
                        for(int i=0;i<5;i++){
                            person.eat();
                        }
                    }
                }
                )
        .start();

        new Thread(
                new Runnable() {
                    @Override
                    public void run() {
                        for(int i=0;i<5;i++){
                            person.drink();
                        }
                    }
                }
                )
        .start();

        for(int i=0;i<10;i++){
            person.play();
        }
    }
}

结果输出:

eat...
eat...
play...
play...
play...
play...
play...
play...
play...
drink...
drink...
play...
play...
play...
play...

以上案例为传统线程调用,打印的结果也是有cpu随机调度输出

现在的需求是:person 吃喝玩方法要有顺序性调用,阻止cpu随机调度输出

参考以下案例(加入Lock实现同步通信):

public class Person {

    Lock lock = new ReentrantLock();
    Condition condition1 = lock.newCondition();
    Condition condition2 = lock.newCondition();
    Condition condition3 = lock.newCondition();
    private int shouldSub = 1;

    public void eat(){
        lock.lock();
        try{
            while(shouldSub!=1){      //是1的时候执行  不是1的时候等待
                try {
                    condition1.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            for(int i=0;i<3;i++){
                System.out.println("eat...");
            }
            shouldSub=2;
            condition2.signal();    // 1 执行完之后 通知2执行
         }finally{
             System.out.println("=======================");
             lock.unlock();
         }
    }

    public void drink(){
        lock.lock();
        try{
            while(shouldSub!=2){ //是2的时候执行  不是2的时候等待
                try {
                    condition2.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            for(int i=0;i<3;i++){
                System.out.println("drink...");
            }
            shouldSub = 3;
            condition3.signal();// 2 执行完之后 通知3执行
         }finally{
             System.out.println("=======================");
             lock.unlock();
         }
    }

    public void play(){
        lock.lock();
        try{
            while(shouldSub!=3){    //是3的时候执行  不是3的时候等待
                try {
                    condition3.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            for(int i=0;i<3;i++){
                System.out.println("play...");
            }
            shouldSub = 1;
            condition1.signal();// 3 执行完之后 通知1执行
        }finally{
            System.out.println("=======================");
             lock.unlock();
         }
    }
}
public class LockConditionTest {
    public static void main(String[] args) {
        final Person person = new Person();
        new Thread(
                new Runnable() {
                    @Override
                    public void run() {
                        for(int i=0;i<5;i++){
                            person.eat();
                        }
                    }
                }
                )
        .start();

        new Thread(
                new Runnable() {
                    @Override
                    public void run() {
                        for(int i=0;i<5;i++){
                            person.drink();
                        }
                    }
                }
                )
        .start();

        for(int i=0;i<5;i++){
            person.play();
        }
    }
}

结果输出:

eat...
eat...
eat...
=======================
drink...
drink...
drink...
=======================
play...
play...
play...
=======================
eat...
eat...
eat...
=======================
drink...
drink...
drink...
=======================
play...
play...
play...
=======================
eat...
eat...
eat...
=======================
drink...
drink...
drink...
=======================
play...
play...
play...
=======================

JAVA线程同步通信的更多相关文章

  1. Java线程并发:知识点

    Java线程并发:知识点   发布:一个对象是使它能够被当前范围之外的代码所引用: 常见形式:将对象的的引用存储到公共静态域:非私有方法中返回引用:发布内部类实例,包含引用.   逃逸:在对象尚未准备 ...

  2. Java线程的概念

    1.      计算机系统 使用高速缓存来作为内存与处理器之间的缓冲,将运算需要用到的数据复制到缓存中,让计算能快速进行:当运算结束后再从缓存同步回内存之中,这样处理器就无需等待缓慢的内存读写了. 缓 ...

  3. Java 线程池框架核心代码分析--转

    原文地址:http://www.codeceo.com/article/java-thread-pool-kernal.html 前言 多线程编程中,为每个任务分配一个线程是不现实的,线程创建的开销和 ...

  4. 细说进程五种状态的生老病死——双胞胎兄弟Java线程

    java线程的五种状态其实要真正高清,只需要明白计算机操作系统中进程的知识,原理都是相同的. 系统根据PCB结构中的状态值控制进程. 单CPU系统中,任一时刻处于执行状态的进程只有一个. 进程的五种状 ...

  5. 【转载】 Java线程面试题 Top 50

    Java线程面试题 Top 50 不管你是新程序员还是老手,你一定在面试中遇到过有关线程的问题.Java语言一个重要的特点就是内置了对并发的支持,让Java大受企业和程序员 的欢迎.大多数待遇丰厚的J ...

  6. 第24章 java线程(3)-线程的生命周期

    java线程(3)-线程的生命周期 1.两种生命周期流转图 ** 生命周期:**一个事物冲从出生的那一刻开始到最终死亡中间的过程 在事物的漫长的生命周期过程中,总会经历不同的状态(婴儿状态/青少年状态 ...

  7. 第23章 java线程通信——生产者/消费者模型案例

    第23章 java线程通信--生产者/消费者模型案例 1.案例: package com.rocco; /** * 生产者消费者问题,涉及到几个类 * 第一,这个问题本身就是一个类,即主类 * 第二, ...

  8. 第22章 java线程(2)-线程同步

    java线程(2)-线程同步 本节主要是在前面吃苹果的基础上发现问题,然后提出三种解决方式 1.线程不安全问题 什么叫线程不安全呢 即当多线程并发访问同一个资源对象的时候,可能出现不安全的问题 对于前 ...

  9. 第21章 java线程(1)-线程初步

    java线程(1)-线程初步 1.并行和并发 并行和并发是即相似又有区别: 并行:指两个或者多个事件在同一时刻点发生. 并发:指两个或多个事件在同一时间段内发生 在操作系统中,并发性是指在一段事件内宏 ...

随机推荐

  1. markdown简介及基本语法

    一.前言 Markdown是一种轻量级标记语言,它以纯文本形式(易读.易写.易更改)编写文档,并最终以HTML格式发布. Markdown也可以理解为将以MARKDOWN语法编写的语言转换成HTML内 ...

  2. jQuery——超链接提示

    在熟悉jQuery过程中,练习超链接提示显示,发现书本上有个问题,经过查询资料,修改如下: <!DOCTYPE html> <html> <head> <ti ...

  3. Tomcat 保存镜像实战操作( 目录挂载方法 )

    查看数据保存的位置 docker inspect --format='{{.Mounts}}' mxg_tomcat 宿主机数据保存在 /usr/local/project , 将此路径数据备份在 b ...

  4. 离线下载Express 2015 for Windows 10

    我在微软https://www.visualstudio.com/zh-cn/downloads/download-visual-studio-vs 点Express 2015 for Windows ...

  5. Linux系统启动过程浅析

    经过老师的讲解以及查阅资料后,现对Linux系统启动做以浅析,仅是个人理解. 主要的步骤有以下几步: 第一步:Power On.用户按下电源开关的那一瞬间,叫Power On阶段 .在这个阶段,BIO ...

  6. Redis分布式锁【实战】

    概述 目前几乎很多大型网站及应用都是分布式部署的,分布式场景中的数据一致性问题一直是一个比较重要的话题.分布式的CAP理论告诉我们“任何一个分布式系统都无法同时满足一致性(Consistency).可 ...

  7. UML建模重点圈划

    面向对象的特征 *P9*>封装性>继承性>多态性>传递性 建模语言的三个类别 *P14*> - 非形式化的.半形式化的和形式化的 UML 特点*15*主要有三个特点:&g ...

  8. 如何从word中复制内容到网站后台编辑器中

    word图片转存,是指UEditor为了解决用户从word中复制了一篇图文混排的文章粘贴到编辑器之后,word文章中的图片数据无法显示在编辑器中,也无法提交到服务器上的问题而开发的一个操作简便的图片转 ...

  9. Leetcode_131. Palindrome Partitioning_[DFS]

    题目链接 Given a string s, partition s such that every substring of the partition is a palindrome. Retur ...

  10. 一种算法的实现,几个相同大小的div组合在一起,判断是不是矩形