body, table{font-family: 微软雅黑; font-size: 10pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}

线程间通信
█ 针对同一个资源的操作有不同的线程
      ☞ 举例:买点影票有卖出的,也有退票的
█ Demo
      ☞ 通过生产线程 和消费线程,对一个生产对象进行操作。

等待唤醒机制
█ Object

>Wait()       // 执行wait会自动释放锁     
>Notify()    //会唤醒其他等待进程,但是不会释放锁;通知其他进程可以执行,其他进程获得可以执行的一个条件进入同步阻塞,当进程获得锁的时候就进入就绪状态    
>notifyAll()

为什么不定义在Thread类中?
//因为锁可以是任意对象。若以线程间通信机制的api必须所有对象都有

线程状态转换图
Baozi.java TestMain.java
package com.java.thread_PV;
public class Baozi {
        private String name;
        private String price;
        private boolean flag;    //flag 为false的时候表示没有数据,为true表示有数据;不初始化默认为FALSE
        public String getName() {
                return name;
        }
        public void setName(String name) {
                this.name = name;
        }
        public String getPrice() {
                return price;
        }
        public void setPrice(String price) {
                this.price = price;
        }
        public Baozi(String name, String price) {
                super();
                this.name = name;
                this.price = price;
        }
        public boolean isFlag() {
                return flag;
        }
        public void setFlag(boolean flag) {
                this.flag = flag;
        }
}

package com.java.thread_PV;
public class TestMain {
        public static void main(String[] args) {
                Baozi baozi=new Baozi(null,null);    //同一个对象,后面确保同一把锁
                Producer producer=new Producer(baozi);
                Customer customer=new Customer(baozi);
                producer.start();
                customer.start();
        }
}

Producer.java Customer.java
package com.java.thread_PV;
public class Producer extends Thread {
        Baozi baozi;
        public Producer(Baozi baozi2) {
                baozi = baozi2;
        }
        public void run() {
                super.run();
                int i = 0;
                while (true) {
                        synchronized (baozi) {  //notify 和  wait 机制需要放到同步代码快里使用
                                if (baozi.isFlag()) {  // flag 为true表示有包子
                                        // 等待 (放弃执行权,让其他线程执行。消费者去消费)
                                        try {
                                                baozi.wait();
                                        } catch (InterruptedException e) {
                                                e.printStackTrace();
                                        }
                                }
                                // 没有包子就生产
                                if (i % 2 == 0) {
                                        baozi.setName("天津狗不理");
                                        baozi.setPrice("10RMB");
                                } else {
                                        baozi.setName("无锡灌汤包");
                                        baozi.setPrice("5RMB");
                                }
                                i++;
                                // 生产完之后?改变标记flag为true,通知消费者来消费
                                baozi.setFlag(true);
                                baozi.notify();   // 通知消费者,但是消费者并不能立即执行;要等后面的代码全部执行完释放锁后,消费者进程抢到锁才能执行
                        }
                }
        }
}
package com.java.thread_PV;
public class Customer extends Thread {
        Baozi baozi;
        public Customer(Baozi baozi2) {
                baozi = baozi2;
        }
        public void run() {
                super.run();
                while (true) {
                        synchronized (baozi) {
                                if (!baozi.isFlag()) { // false 表示没有包子,就等待
                                        try {
                                                baozi.wait();  //等待的时候放弃cpu执行权。。但是这里要释放锁!!这里wait方法会自动释放锁,所以不会形成死锁
                                        } catch (InterruptedException e) {
                                                e.printStackTrace();
                                        }
                                }
                                // 如果有就消费
                                System.out.println("Customer.consume()" + baozi.getName() + " :" + baozi.getPrice());
                                // 消费完就修改标记,通知生产者去生产
                                baozi.setFlag(false);
                                baozi.notify();  // 通知生产者,notify不会释放锁,如果下面还有代码,这里还会接着执行,不会释放锁;另外的进程不会拿到锁,会进入同步阻塞;拿到锁就进入就绪状态
                        }
                }
        }
}

Java——线程间通信的更多相关文章

  1. Java线程间通信-回调的实现方式

    Java线程间通信-回调的实现方式   Java线程间通信是非常复杂的问题的.线程间通信问题本质上是如何将与线程相关的变量或者对象传递给别的线程,从而实现交互.   比如举一个简单例子,有一个多线程的 ...

  2. Java线程间通信之wait/notify

    Java中的wait/notify/notifyAll可用来实现线程间通信,是Object类的方法,这三个方法都是native方法,是平台相关的,常用来实现生产者/消费者模式.我们来看下相关定义: w ...

  3. java线程间通信:一个小Demo完全搞懂

    版权声明:本文出自汪磊的博客,转载请务必注明出处. Java线程系列文章只是自己知识的总结梳理,都是最基础的玩意,已经掌握熟练的可以绕过. 一.从一个小Demo说起 上篇我们聊到了Java多线程的同步 ...

  4. 说说Java线程间通信

    序言 正文 [一] Java线程间如何通信? 线程间通信的目标是使线程间能够互相发送信号,包括如下几种方式: 1.通过共享对象通信 线程间发送信号的一个简单方式是在共享对象的变量里设置信号值:线程A在 ...

  5. 说说 Java 线程间通信

    序言 正文 一.Java线程间如何通信? 线程间通信的目标是使线程间能够互相发送信号,包括如下几种方式: 1.通过共享对象通信 线程间发送信号的一个简单方式是在共享对象的变量里设置信号值:线程A在一个 ...

  6. java线程间通信1--简单实例

    线程通信 一.线程间通信的条件 1.两个以上的线程访问同一块内存 2.线程同步,关键字 synchronized 二.线程间通信主要涉及的方法 wait(); ----> 用于阻塞进程 noti ...

  7. java线程间通信之通过管道进行通信

    管道流PipeStream是一种特殊的流,用于在不同线程间直接传送数据,而不需要借助临时文件之类的东西. jdk中提供了四个类来使线程间可以通信: 1)PipedInputStream和PipedOu ...

  8. Java——线程间通信问题

     wait和sleep区别: 1.wait可以指定时间可以不指定.     sleep必须指定时间. 2.在同步时,对cpu的执行权和锁的处理不同.     wait:释放执行权,释放锁.     ...

  9. Java线程间通信

    1.由来 当需要实现有顺序的执行多个线程的时候,就需要进行线程通信来保证 2.实现线程通信的方法 wait()方法: wait()方法:挂起当前线程,并释放共享资源的锁 notify()方法: not ...

随机推荐

  1. ElasticSearch 5.4 自定义插件

    ElasticSearch 做为数据仓库处理速度确实很强,但是很多和业务相关的函数ElasticSearch怎么支持的,通过查询发现,ElasticSearch支持自定义插件(相当于自定义函数),通过 ...

  2. 3G下的无压缩视频传输(基于嵌入式linux) (转载)

    本课题研究嵌入式系统在数据采集,3G无线通信方面的应用,开发集视频采集.地理信息采集.无线传输.客户机/服务器模式于一体的车载终端,实现终端采集视频与GPS信息的传输,支持服务器端显示视频与GPS信息 ...

  3. Springboot统一参数验证方式

    Springboot统一验证方式 在提供http api 接口形式的服务中,通过都会传递参数为一个对象.我们需要对这个对象的各个字段进行校验.来判断是否为合法值. 传统的方式为自己获取每个字段的值,自 ...

  4. API接口自动化之3 同一个war包中多个接口做自动化测试

    同一个war包中多个接口做自动化测试 一个接口用一个测试类,每个测试用例如下,比如下面是4个测试用例,每个详细的测试用例中含有请求入参,返回体校验,以此来判断每条测试用例是否通过 一个war包中,若含 ...

  5. pandas (loc、iloc、ix)的区别

    loc:通过行标签索引数据 iloc:通过行号索引行数据 ix:通过行标签或行号索引数据(基于loc和iloc的混合) 使用loc.iloc.ix索引第一行数据: loc: iloc: ix:

  6. re.sub

    1.re.sub主要功能实现正则的替换. re.sub定义: sub(pattern, repl, string, count=0, flags=0) 意思为:对字符串string按照正则表达式pat ...

  7. 分析 HTML 代码并提取数据

    在前面的内容中,我们已经学习了 HTML.CSS 和 XPath 的基础知识.从真实世界的网页中获取数据,关键在于如何编写合适的 CSS 或者 XPath 选择器.本节介绍一些确定选择器的简单方法.假 ...

  8. Android ListView常见配置说明

    ListView是我们经常使用的控件,但是使用中却因为各种原因无法设置出我们需要的效果,现将常用的设置记录下来方便以后查询. 1.拖动时背景变黑 android:cacheColorHint=&quo ...

  9. 你的centos/linux下有多个php.ini,不确定是哪个时

    你的centos/linux下有多个php.ini,不确定是哪个时,但是你自己知道,你的php安装目录. 比如我的php安装目录是 /usr/local/php 那么可以通过命令来查找php.ini的 ...

  10. jsp post/get中接处理

    jsp post/get中接处理 以参数:username为便 post接收中文比get接收中文要方便多了. <%@ page contentType="text/html;chars ...