状态是针对当前的线程调度的情况进行描述的。
线程是调度的基本单位,状态是线程的属性。
4~6三个状态都是阻塞状态。(都是表示线程PCB正在阻塞队列中)只不过是不同原因的阻塞。
TERMINATED状态中,内核中线程的PCB被释放了,此时代码中的t对象也就没用了。Java中对象的生命周期自有其规则,这个生命周期和系统内核中的线程并非完全一致,**内核的线程释放的时候,无法保证Java代码中的t对象也立即释放。**此时t对象标识为:无效。虽然t对象无效了,但是t对象依旧可以完成调用函数等操作。
一个线程只能start一次。
public class ThreadDemo10 {
public static void main(String[] args) throws InterruptedException {
Thread t = new Thread(()->{
for (int i = 0; i < 100; i++) {
for (int j = 0; j < 1000_0000; j++) {
int a = 10;
a += 10;
}
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
});
//未start之前是new状态
System.out.println("start之前:"+t.getState());
t.start();
//t执行中的状态runable
for (int i = 0; i < 1000; i++) {
System.out.println("t 执行中的状态: " + t.getState());
}
t.join();
// 线程执行完毕之后, 就是 TERMINATED 状态
System.out.println("t 结束之后: " + t.getState());
}
}
多线程的意义:
多线程可以更充分利用多核心的CPU资源,从而加快程序的运行效率。
线程安全的问题的根本原因就是抢占式执行,带来的随机性。
我们来看一个例子:
class Counter {
public int count = 0;
public void add() {
count++;
}
}
public class ThreadDemo12 {
public static void main(String[] args) {
Counter counter = new Counter();
// 搞两个线程, 两个线程分别针对 counter 来 调用 5w 次的 add 方法
Thread t1 = new Thread(() -> {
for (int i = 0; i < 50000; i++) {
counter.add();
}
});
Thread t2 = new Thread(() -> {
for (int i = 0; i < 50000; i++) {
counter.add();
}
});
// 启动线程
t1.start();
t2.start();
// 等待两个线程结束
try {
t1.join();
t2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
// 打印最终的 count 值
System.out.println("count = " + counter.count);
}
}
以上代码我们预期结果是100000次,但是运行结果如下:
为什么出现这个bug呢?
count++;本质上在操作系统中分成三 步:
当两个线程并发执行count++时,就相当于load add save同时执行。此时就会产生结果上的差异。
可能执行的方式:
箭头是时间轴,靠上就是先执行,靠下就是后执行。
由于线程之间是随机调度的,导致此处的调度顺序充满其他可能性。
要想解决线程安全问题,主要手段就是从原子性入手,把非原子的操作,变成原子的。加锁。
上面我们说到。通过加锁,我们可以把不是原子的,转成原子的。
加了synchronized之后,进入add方法就会加锁,出了add方法就会解锁。
如果两个线程同时尝试加锁,此时只有一个能获取锁成功,另一个只能阻塞等待(BLOCK)。一直阻塞到刚才的线程释放锁(解锁),当前线程才能加锁成功。
加锁,说是保证原子性,但并不是说让这里的三个操作一次性完成,也不是这三步操作过程中不进行调度,而是让其他也想操作的线程阻塞等待。加锁本质上是把并发变成了并行。
加锁操作会影响程序的速率,在实际过程中我们要通过实际情景来对其进行合理加锁。
synchronized使用方法:
- 修饰方法
(1)修饰普通方法(锁对象是this)
(2)修饰静态方法(锁对象是类对象(Counter.class))- 修饰代码块(显示/手动指定锁对象)
加锁,要明确执行对哪个对象进行加锁的。
如果两个线程针对同一个对象加锁,会产生阻塞等待。(锁竞争/锁冲突)
如果两个对象针对不同对象加锁,不会参数阻塞等待。(不会锁竞争/锁冲突)
一定要注意锁对象是哪个!
监视器锁也就是synchronized。有时会在异常中看到这个词。
理解 “阻塞等待”: 针对每一把锁, 操作系统内部都维护了一个等待队列. 当这个锁被某个线程占有的时候, 其他线程尝 试进行加锁,就加不上了, 就会阻塞等待, 一直等到之前的线程解锁之后, 由操作系统唤醒一个新的 线程, 再来获取到这个锁。
注意:
上一个线程解锁之后, 下一个线程并不是立即就能获取到锁. 而是要靠操作系统来 “唤醒”. 这
也就是操作系统线程调度的一部分工作.
假设有 A B C 三个线程, 线程 A 先获取到锁, 然后 B 尝试获取锁, 然后 C 再尝试获取锁, 此时 B和 C 都在阻塞队列中排队等待. 但是当 A 释放锁之后, 虽然 B 比 C 先来的, 但是 B 不一定就能获取到锁, 而是和 C 重新竞争, 并不遵守先来后到的规则.
程序中一旦出现死锁,就会导致线程崩溃了(无法继续执行后续工作)。程序就会产生严重的bug。死锁一般是非常隐蔽的。
public class ThreadDemo13 {
public static void main(String[] args) {
Object yingyu = new Object();
Object shuxue = new Object();
Thread xiaohong = new Thread(() -> {
synchronized (yingyu) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (shuxue) {
System.out.println("小红抄到了小兰的数学作业!小红写完了所有作业。");
}
}
});
Thread xiaolan = new Thread(() -> {
synchronized (shuxue) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (yingyu) {
System.out.println("小兰抄到了小红的英语作业!小兰写完了所有作业。");
}
}
});
xiaohong.start();
xiaolan.start();
}
}
通过运行结果可知,此时没有线程拿到两把锁。
BLOCK表示获取锁,获取不到阻塞状态。
java:16行
java:32
针对这样的死锁问题,需要借助jconsole这样的工作来进行定位,看线程的状态和调用栈。就可以分析代码再哪里死锁了。
public class ThreadDemo13 {
public static void main(String[] args) {
Object yingyu = new Object();
Object shuxue = new Object();
Thread xiaohong = new Thread(() -> {
// 假设 yingyu 是 1 号, shuxue 是 2 号, 约定先拿小的, 后拿大的.
synchronized (yingyu) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (shuxue) {
System.out.println("小红抄到了小兰的数学作业!小红写完了所有作业。");
}
}
});
Thread xiaolan = new Thread(() -> {
synchronized (yingyu) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (shuxue) {
System.out.println("小兰抄到了小红的英语作业!小兰写完了所有作业。");
}
}
});
xiaohong.start();
xiaolan.start();
}
}
如何避免死锁?(以循环等待为突破口)
方法:给锁编号,然后指定一个固定的顺序(从小到大)来加锁,任意线程加多把锁的时候,都让线程遵守上述顺序,此时循环等待自然破除。这是解决死锁最简单可靠的办法。
Java 标准库中很多都是线程不安全的. 这些类可能会涉及到多线程修改共享数据, 又没有任何加锁措施.
但是还有一些是线程安全的. 使用了一些锁机制来控制.
还有的虽然没有加锁, 但是不涉及 “修改”, 仍然是线程安全的