一、概述

1.1 什么是多线程之间通讯?

  • 多线程之间通讯,其实就是多个线程在操作同一个资源,但是操作的动作不同。

1.2 案例

  • 需求:第一个线程写入(input)用户,另一个线程取读取(out)用户。实现读一个,写一个操作。

代码实现

/**
* 测试等待/通知机制
* @author hao
*
*/
public class Test_NoWaitNotify {
public static void main(String[] args) {
Res res = new Res();
IntThrad intThrad = new IntThrad(res);
OutThread outThread = new OutThread(res);
intThrad.start();
outThread.start();
}
} /*
* 共享资源源实体类
*
*/
class Res {
public String userSex;
public String userName;
} /*
*
* 输入线程资源
*
*/ class IntThrad extends Thread {
private Res res; public IntThrad(Res res) {
this.res = res;
} @Override
public void run() {
int count = 0;
while (true) {
if (count == 0) {
res.userName = "小明";
res.userSex = "男";
} else {
res.userName = "小红";
res.userSex = "女";
}
count = (count + 1) % 2;
}
}
} /*
*
* 输出线程
*
*/ class OutThread extends Thread {
private Res res; public OutThread(Res res) {
this.res = res;
} @Override
public void run() {
while (true) {
System.out.println(res.userName + "--" + res.userSex);
}
}
}
  • 改例中数据发生错乱,造成线程安全问题

解决线程安全问题

加入synchronized 关键字

public class Test002 {
public static void main(String[] args) {
Res2 res = new Res2();
IntThrad2 intThrad = new IntThrad2(res);
OutThread2 outThread = new OutThread2(res);
intThrad.start();
outThread.start();
}
} /*
* 共享资源源实体类
*
*/
class Res2 {
public String userSex;
public String userName;
} /*
*
* 输入线程资源
*
*/ class IntThrad2 extends Thread {
private Res2 res; public IntThrad2(Res2 res) {
this.res = res;
} @Override
public void run() {
int count = 0;
while (true) {
synchronized (res) {
if (count == 0) {
res.userName = "小明";
res.userSex = "男";
} else {
res.userName = "小红";
res.userSex = "女";
}
count = (count + 1) % 2;
}
}
}
} /*
*
* 输出线程
*
*/ class OutThread2 extends Thread {
private Res2 res; public OutThread2(Res2 res) {
this.res = res;
} @Override
public void run() { while (true) {
synchronized (res) {
System.out.println(res.userName + "--" + res.userSex);
}
}
}
}

二、等待通知机制

2.1 示例

  • 上面例子中解决了线程安全问题,但是如果我们要求读一个写一个,那么该如何解决
  • 1.因为涉及到对象锁,他们必须都放在synchronized中来使用. Wait、Notify一定要在synchronized里面进行使用。
  • 2.Wait必须暂定当前正在执行的线程,并释放资源锁,让其他线程可以有机会运行
  • 3.notify/notifyall: 唤醒因锁池中的线程,使之运行
  • 注意:一定要在线程同步中使用,并且是同一个锁的资源
class Res3 {
public String userSex;
public String userName;
// 线程通讯标识
public boolean flag = false;
} class IntThrad3 extends Thread {
private Res3 res; public IntThrad3(Res3 res) {
this.res = res;
} @Override
public void run() {
int count = 0;
while (true) {
synchronized (res) {
if (res.flag) {
try {
// 当前线程变为等待,但是可以释放锁
res.wait();
} catch (Exception e) { }
}
if (count == 0) {
res.userName = "小明";
res.userSex = "男";
} else {
res.userName = "小红";
res.userSex = "女";
}
count = (count + 1) % 2;
res.flag = true;
// 唤醒当前线程
res.notify();
} }
}
} class OutThread3 extends Thread {
private Res3 res; public OutThread3(Res3 res) {
this.res = res;
} @Override
public void run() {
while (true) {
synchronized (res) {
if (!res.flag) {
try {
res.wait();
} catch (Exception e) {
// TODO: handle exception
}
}
System.out.println(res.userName + "--" + res.userSex);
res.flag = false;
res.notify();
}
}
}
} public class Test003 {
public static void main(String[] args) {
Res3 res = new Res3();
IntThrad3 intThrad = new IntThrad3(res);
OutThread3 outThread = new OutThread3(res);
intThrad.start();
outThread.start();
}
}

2.2 wait与sleep区别

  • 对于sleep()方法,我们首先要知道该方法是属于Thread类中的。而wait()方法,则是属于Object类中的。
  • sleep()方法导致了程序暂停执行指定的时间,让出cpu该其他线程,但是他的监控状态依然保持着,当指定的时间到了又会自动恢复运行状态。
  • 在调用sleep()方法的过程中,线程不会释放对象锁。而当调用wait()方法的时候,线程会放弃对象锁,进入等待此对象的线程等待池,只有针对此对象调用notify()方法后本线程才进入对象锁定池准备,获取对象锁进入运行状态。

三、Lock锁

3.1 概述

  • 在 jdk1.5 之后,并发包中新增了 Lock 接口(以及相关实现类)用来实现锁功能,Lock 接口提供了与 synchronized 关键字类似的同步功能,但需要在使用时手动获取锁和释放锁。

  • Lock写法,基本用法

Lock lock  = new ReentrantLock();
lock.lock();
try{
//可能会出现线程安全的操作
}finally{
//一定在finally中释放锁
//也不能把获取锁在try中进行,因为有可能在获取锁的时候抛出异常
lock.ublock();
}

3.2 等待/通知机制(Condition)

  • Condition的功能类似于在传统的线程技术中的,Object.wait()和Object.notify()的功能。

  • 代码示例,将上面将的例子改造如下

public class TL002_Condition {
public static void main(String[] args) throws InterruptedException { Res res = new Res();
IntThread intThread = new IntThread(res);
OutThread outThread = new OutThread(res);
intThread.start();
outThread.start();
}
} /*
* 共享资源
*/
class Res { public String name;
public String sex;
public boolean flag = false;
public Lock lock = new ReentrantLock();
public Condition condition=lock.newCondition();
} /*
* 写入线程
*/
class IntThread extends Thread {
public Res res; public IntThread(Res res) {
this.res = res;
} @Override
public void run() {
int count = 0; // 1
while (true) {
try {
res.lock.lock();
if(res.flag){
res.condition.await();
}
if (count == 0) {
res.name = "小明";
res.sex = "男";
} else {
res.name = "小红";
res.sex = "女";
}
count = (count + 1) % 2;
res.flag=true;
res.condition.signal();
} catch (Exception e) {
} finally {
res.lock.unlock();
} } } } // 读取线程
class OutThread extends Thread { public Res res; public OutThread(Res res) {
this.res = res;
} @Override
public void run() {
while (true) { try {
res.lock.lock();
if(!res.flag){
res.condition.await();
}
Thread.sleep(1000);
System.out.println(res.name + "," + res.sex);
res.flag = false;
res.condition.signal();
} catch (Exception e) {
// TODO: handle exception
} finally {
res.lock.unlock();
}
} }
}

3.3 Lock与synchronized 比较

  1. 锁的实现:synchronized 是 JVM 实现的,而 ReentrantLock 是 JDK 实现的。
  2. 性能:新版本 Java 对 synchronized 进行了很多优化,例如自旋锁等,synchronized 与 ReentrantLock 大致相同。
  3. 等待可中断:当持有锁的线程长期不释放锁的时候,正在等待的线程可以选择放弃等待,改为处理其他事情。ReentrantLock 可中断,而 synchronized 不行。
  4. 公平锁,公平锁是指多个线程在等待同一个锁时,必须按照申请锁的时间顺序来依次获得锁。synchronized 中的锁是非公平的,ReentrantLock 默认情况下也是非公平的,但是也可以是公平的。
  5. 锁绑定多个条件:一个 ReentrantLock 可以同时绑定多个 Condition 对象。
  6. 使用选择
    • 除非需要使用 ReentrantLock 的高级功能,否则优先使用 synchronized。这是因为 synchronized 是 JVM 实现的一种锁机制,JVM 原生地支持它,而 ReentrantLock 不是所有的 JDK 版本都支持。并且使用 synchronized 不用担心没有释放锁而导致死锁问题,因为 JVM 会确保锁的释放。

【Java并发】线程通信的更多相关文章

  1. Java 并发 线程同步

    Java 并发 线程同步 @author ixenos 同步 1.异步线程本身包含了执行时需要的数据和方法,不需要外部提供的资源和方法,在执行时也不关心与其并发执行的其他线程的状态和行为 2.然而,大 ...

  2. Java 并发 线程的优先级

    Java 并发 线程的优先级 @author ixenos 低优先级线程的执行时刻 1.在任意时刻,当有多个线程处于可运行状态时,运行系统总是挑选一个优先级最高的线程执行,只有当线程停止.退出或者由于 ...

  3. Java 并发 线程属性

    Java 并发 线程属性 @author ixenos 线程优先级 1.每当线程调度器有机会选择新线程时,首先选择具有较高优先级的线程 2.默认情况下,一个线程继承它的父线程的优先级 当在一个运行的线 ...

  4. Java 并发 线程的生命周期

    Java 并发 线程的生命周期 @author ixenos 线程的生命周期 线程状态: a)     New 新建 b)     Runnable 可运行 c)     Running 运行 (调用 ...

  5. Java并发——线程安全、线程同步、线程通信

    线程安全 进程间"共享"对象 多个“写”线程同时访问对象. 例:Timer实例的num成员,即add()方法是用的次数.即Timer实例是资源对象. class TestSync ...

  6. java多线程-线程通信

    线程通信的目标是使线程间能够互相发送信号.另一方面,线程通信使线程能够等待其他线程的信号. 通过共享对象通信 忙等待 wait(),notify()和 notifyAll() 丢失的信号 假唤醒 多线 ...

  7. 从JDK源码角度看java并发线程的中断

    线程的定义给我们提供了并发执行多个任务的方式,大多数情况下我们会让每个任务都自行执行结束,这样能保证事务的一致性,但是有时我们希望在任务执行中取消任务,使线程停止.在java中要让线程安全.快速.可靠 ...

  8. Java并发——线程介绍

    前言: 互联网时代已经发展到了现在.从以前只考虑小流量到现在不得不去考虑高并发的问题.扯到了高并发的问题就要扯到线程的问题.你是否问过自己,你真正了解线程吗?还是你只知道一些其他博客里写的使用方法.下 ...

  9. java并发线程池---了解ThreadPoolExecutor就够了

    总结:线程池的特点是,在线程的数量=corePoolSize后,仅任务队列满了之后,才会从任务队列中取出一个任务,然后构造一个新的线程,循环往复直到线程数量达到maximumPoolSize执行拒绝策 ...

  10. java 并发——线程

    一.前言 前一篇文章总结了对 java 并发中的内置锁的理解,这篇文章来说说线程 ,并发与线程总有剪不断理还乱的关系.关于 java 线程的基本概念.线程与进程的关系以及如何创建线程,想必大家都很清楚 ...

随机推荐

  1. 图解 HTTP 笔记(二)——简单的 HTTP 协议

    本章主要以 HTTP 1.0 为例,讲解 HTTP 协议的基本结构. 在两台计算机之间使用 HTTP 协议进行通讯时,在一条通讯线路上必定有一端是客户端,另一端则是服务器端. 请求访问文本或图像等资源 ...

  2. python抽取指定url页面的title方法

    python抽取指定url页面的title方法 今天简单使用了一下python的re模块和lxml模块,分别利用的它们提供的正则表达式和xpath来解析页面源码从中提取所需的title,xpath在完 ...

  3. JAVA 基础编程练习题45 【程序 45 被 9 整除】

    45 [程序 45 被 9 整除] 题目:判断一个素数能被几个 9 整除 package cskaoyan; public class cskaoyan45 { public static void ...

  4. Anaconda+tensorflow(不用创建虚拟环境)

    网上大部分教程都是:创建tensorflow虚拟环境(conda create -n tensorflow python=3.6),然后在虚拟环境中pip install tensorflow,但是每 ...

  5. STM32命名规则解析

  6. webdriervAPI(多表单切换)

    讲三个方法 driver.switch_to.frame("第一个iframe标签属性值") driver.switch_to.frame(" 第二个iframe标签属性 ...

  7. linux中文件权限的字母含义

    一.文件系统部分 - 普通文件(文本文件,二进制文件,压缩文件,电影,图片...)d 目录文件(蓝色)b 设备文件(块设备)存储设备硬盘,U盘 /dev/sda, /dev/sda1c 设备文件(字符 ...

  8. Leetcode之动态规划(DP)专题-647. 回文子串(Palindromic Substrings)

    Leetcode之动态规划(DP)专题-647. 回文子串(Palindromic Substrings) 给定一个字符串,你的任务是计算这个字符串中有多少个回文子串. 具有不同开始位置或结束位置的子 ...

  9. HDU3191 【输出次短路条数】

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3191 How Many Paths Are There Time Limit: 2000/1000 M ...

  10. There are no packages available

    { "bootstrapped": true, "channels": [ "https://raw.githubusercontent.com/Ja ...