如果要问我Java当中最难的部分是什么?最有意思的部分是什么?最多人讨论的部分是什么?那我会毫不犹豫地说:多线程。

Java多线程说它难,也不难,就是有点绕;说它简单,也不简单,需要理解的概念很多,尤其是很多底层知识,如数据结构、操作系统的部分。

Java多线程掌握得好,不仅仅只是对Java,对任何其他具有并发特性的编程语言,甚至是操作系统,都能有更全面和准确的认识。

Java多线程最大的特点,而且也是唯一确定的一件事,那就是:在多线程环境下,程序的运行结果是无法预料的,但这也正是它最有趣的地方。

在了解多线程之前,最好先知道什么是并发,什么是并行。不然很容易迷糊。

总的来说,就是这样:

并行:同一时刻可以同时发生/执行多个任务。

并发:同一时刻只能发生/执行一个任务。


学习多线程最好从如下六个方面循序渐进(纯粹个人经验和建议,可无视):

1、线程生命周期:NEW、RUNNABLE(READY、RUNNING)、BLOCKED、WAITING、TIMED_WAITING、TERMINATED状态

2、关键字:synchronized和volatile

3、线程池:ThreadPoolExecutor

4、锁(AQS):悲观锁/乐观锁、轻量级锁/重量级锁、自旋锁/可重入锁等各种锁

5、CAS:各种原子类

6、并发工具类:ArrayBlockingQueue、CountDownLatch、CyclicBarrier、Semaphore等


Java多线程用一句话总结就是「6类5法」。

所谓「6类」,就是多状态的状态分为这6类:

1、新建(NEW):新创建了一个线程,但还没调用start方法

2、运行(RUNNABLE)

2.1、就绪(ready):运行start方法后,线程位于可运行线程池中,等待被调度

2.2、运行中(RUNNING):就绪的线程获得CPU的时间片就变为运行中

3、阻塞(BLOCKED):线程等待获取锁

4、等待(WAITING):接收事件通知后或系统中断后进入等待

5、超时(TIMED_WAITING):等待指定时间后会自行返回

6、终止(TERMINATED):线程已执行完毕

这是线程生命周期的状态变化图:

简单来说,就是这样:

而所谓「5法」就是线程的核心方法是这么5个:

1、wait:当前线程调用锁对象的wait方法,当前线程释放锁,进入等待状态,由其他线程接着执行

2、notify/notifyAll:唤醒任意一个或全部等待的线程后接着执行,但并不释放锁

3、join:当前线程调用其他线程的join方法,调用后当前线程进入等待状态

4、yield:当前线程调用,调用后暂停执行(可能无效),变为就绪态

5、sleep:当前线程调用,调用后进入TIME_WAITING状态


用代码来解释一下会更直观一些。

第一种wait/notify的情况:

public static void main(String[] args) {
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
System.out.println("t1 start");
try {
// t1释放锁
"锁".wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("t1 end");
}
}
});
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
System.out.println("t2 start");
try {
// 通知t1进入等待队列
"锁".notify();
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("t2 end");
}
}
}); t1.start();
t2.start();
}

此时代码执行流程(两种可能):

1、T1先执行

  1.1、T1启动,wait让出锁,让出CPU,T2获得CPU,T2启动,notify了通过object锁等待的线程

  1.2、T1被唤醒后等待启动,T2继续执行,T2执行完,T1获得CPU后继续执行

2、T2先执行

  2.1、T2执行完,T1启动,让出CPU,由于没有线程再来执行notify,程序无限期等待

这里要强调的重点是:

1、wait会让出CPU而notify不会

2、wait重点在通知其它同用一个object的线程“我暂时不用了”,并且让出CPU

3、notify重点在于通知使用object的对象“我用完了!”

如果说只有两个线程的时候,还能尝试着分析一下结果,那么当有四个线程的时候会如何呢?看看代码:

public static void main(String[] args) {
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
System.out.println("t1 start");
try {
// t1释放锁
"锁".wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("t1 end");
}
}
});
Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
try {
System.out.println("t2 start");
// 随机通知一个等待的线程进入等待队列
"锁".notify();
System.out.println("t2 end");
} catch (Exception e) {
e.printStackTrace();
}
}
}
}); Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
try {
System.out.println("t3 start");
// 随机通知一个等待的线程进入等待队列
"锁".notify();
System.out.println("t3 end");
} catch (Exception e) {
e.printStackTrace();
}
}
}
});
Thread t4 = new Thread(new Runnable() {
@Override
public void run() {
synchronized ("锁") {
try {
System.out.println("t4 start");
// t4释放锁
"锁".wait();
System.out.println("t4 end");
} catch (Exception e) {
e.printStackTrace();
}
}
}
}); t1.start();
t2.start();
t3.start();
t4.start();
}

然后同时开启这四个线程,但结果是无法预料!为什么?因为只有两种可能的流程(要么wait先执行完,要么notify先执行完),至于每种流程里面怎么执行的?不知道!不清楚!无法预料!这就是多线程让人困惑的地方和魅力所在。

而且线程还有一个无赖的行为就是:虽然你有优先级,但我不保证有用!

public class MyThread extends Thread {
MyThread(String s) {
super(s);
} @Override
public void run() {
for (int i = 0; i <= 10; i++) {
System.out.println(getName() + " : " + i);
if (i == 5) {
Thread.yield();
}
}
} public static void main(String[] args) throws InterruptedException {
System.out.println("主线程启动");
Thread t1 = new MyThread("t1");
Thread t2 = new MyThread("t2");
t1.setPriority(Thread.MIN_PRIORITY);
t1.start();
t2.setPriority(Thread.MAX_PRIORITY);
t2.start();
t1.join();
t2.join();
System.out.println("主线程结束");
}
}

这里不管怎么设置t1或者t2的优先级,都没有用,运行的结果每次都可能不一样。

线程的生命周期6类5法算是比较简单的,是基础中的基础。但是用好很难,关键在于多练多想,多多尝试各种组合。

Java多线程-线程生命周期(一)的更多相关文章

  1. Java多线程:生命周期,实现与调度

    Java线程生命周期 Java线程实现方法 继承Thread类,重写run()方法 实现Runnable接口,便于继承其他类 Callable类替换Runnable类,实现返回值 Future接口对任 ...

  2. Java多线程-线程的生命周期

    线程可以分为4个状态:New(新生),Runnable(可运行):为了方便分析,还可将其分为:Runnable与Running.blocked(被阻塞),Dead(死亡). 与人有生老病死一样,线程也 ...

  3. Java多线程——线程的优先级和生命周期

    Java多线程——线程的优先级和生命周期 摘要:本文主要介绍了线程的优先级以及线程有哪些生命周期. 部分内容来自以下博客: https://www.cnblogs.com/sunddenly/p/41 ...

  4. Java多线程与并发——线程生命周期和线程池

    线程生命周期:  线程池:是预先创建线程的一种技术.线程池在还没有任务到来之前,创建一定数量的线程,放入空闲队列中,然后对这些资源进行复用.减少频繁的创建和销毁对象. java里面线程池的顶级接口是E ...

  5. [译]线程生命周期-理解Java中的线程状态

    线程生命周期-理解Java中的线程状态 在多线程编程环境下,理解线程生命周期和线程状态非常重要. 在上一篇教程中,我们已经学习了如何创建java线程:实现Runnable接口或者成为Thread的子类 ...

  6. Java线程生命周期

    当你需要使用Java线程在多线程环境下进行编程时,理解Java的线程周期与线程的状态是非常重要的.通过实现Runnale接口或者继承Thread类,我们可以创建线程,为了启动一个线程,我们需要创建一个 ...

  7. Java线程生命周期与状态切换

    前提 最近有点懒散,没什么比较有深度的产出.刚好想重新研读一下JUC线程池的源码实现,在此之前先深入了解一下Java中的线程实现,包括线程的生命周期.状态切换以及线程的上下文切换等等.编写本文的时候, ...

  8. Java并发编程实战(5)- 线程生命周期

    在这篇文章中,我们来聊一下线程的生命周期. 目录 概述 操作系统中的线程生命周期 Java中的线程生命周期 Java线程状态转换 运行状态和阻塞状态之间的转换 运行状态和无时限等待状态的切换 运行状态 ...

  9. Java并发编程之线程生命周期、守护线程、优先级、关闭和join、sleep、yield、interrupt

    Java并发编程中,其中一个难点是对线程生命周期的理解,和多种线程控制方法.线程沟通方法的灵活运用.这些方法和概念之间彼此联系紧密,共同构成了Java并发编程基石之一. Java线程的生命周期 Jav ...

随机推荐

  1. 436. 寻找右区间--LeetCode_暴力

    来源:力扣(LeetCode) 链接:https://leetcode.cn/problems/find-right-interval 著作权归领扣网络所有.商业转载请联系官方授权,非商业转载请注明出 ...

  2. JCEF 初体验,window系统构建jar包

    前言 本文记录如何通过jcef源代码去构建自己所需要的jar包,此文章构建的为windows64位jcef 的 jar 包,若需要构建 32 位的 jar 包,则需要按照文章将相关准备软件设置为 32 ...

  3. .Net+Vue3实现数据简易导入功能

    在开发的过程中,上传文件或者导入数据是一件很常见的事情,导入数据可以有两种方式: 前端上传文件到后台,后台读取文件内容,进行验证再进行存储 前端读取数据,进行数据验证,然后发送数据到后台进行存储 这两 ...

  4. OSI模型 TCP/IP协议

    常见术语 网络相关的术语 1.拓扑:物理拓扑-----体现了设备之间的连接关系 逻辑拓扑----设备之间的通信关系 2.数据载荷:传递的实际信息 3.报文(PDU--协议数据单元) 4.数据头部的作用 ...

  5. rtmp/rtsp/hls公网测试地址

    相信大家在调试播放器的时候,都有这样的困惑,很难找到合适的公有测试源,以下是大牛直播整理的真正可用的直播地址源. 其中,rtmp和rtsp的url,用https://github.com/daniul ...

  6. KingbaseES V8R6集群管理运维案例之---repmgr standby switchover故障

    案例说明: 在KingbaseES V8R6集群备库执行"repmgr standby switchover"时,切换失败,并且在执行过程中,伴随着"repmr stan ...

  7. Mac_mysql_密码重置

    1 通过Mac 的设置 stop mysql 2 跳过权限认证 // 进入数据库指令文件 cd /usr/local/mysql/bin // 跳过权限认证 sudo ./mysqld_safe -- ...

  8. 果汁 DI 介绍

    Guice (英音同 'juice[果汁]') 是一个为 JDK8 及以上提供的轻量依赖注入框架. 目录 三级标题 三级标题 四级标题 三级标题 三级标题 /** * Animal */ interf ...

  9. 华南理工大学 Python第6章课后测验-2

    1.(单选)以下关于语句 a = [1,2,3,(4,5)]的说法中,正确的个数有( )个.(1)a是元组类型   (2)a是列表类型  (3)a有5个元素      (4)a有4个元素(5)a[1] ...

  10. 算法:KMP, str1字符串是否包含str2字符串

    [普通解法]从左到右遍历str1的每一个字符,然后看如果 以当前字符作为第一个字符出发 是否匹配 str2字符串. [KMP算法] 1)生成一个nextArr数组,长度与str2字符串长度一样.i 的 ...