四种方式
start0() native方法 在JVM源码,java其实是应用层,线程并不是java产生的,而是底层操作系统支持线程,jvm调用操作系统去创建线程,针对不同平台做了封装,在各个操作系统都能创建线程,启动了线程之后会去回调run方法
怎么终止线程
一定不是stop! 这是个过期的方法,挂起和恢复也不推荐使用,
相当于Kill-9 虽然能完成效果,不安全,不优雅
无法知道被关闭的线程的状态,运行到一半 被关闭了。。
Thread.interrupt() 终止才是对的
六种状态
线程池原理
线程工具类四种方法
为何不推荐使用现成的方法
自己创建线程 七大参数含义
阻塞队列
JMM内存模型
硬盘》内存》cpu
cpu计算能力超强,还没给内存,不能一直耗着,或者数据还没传过来, 所以有cpu缓存这个东西
volatile
- 可见性
某一个线程修改了值写回主内存,其他线程马上要获取取通知,你的值不是最新的了,作废,要重新回去拿到最新的值 - 不保证原子性
线程1 拿到值改完之后写回主内存,线程2应该去拿到主内存的值后再去操作,但是!由于多线程竞争的调度关系,两个都准备写,1准备写,突然挂起了,线程2写成了, 去通知的时候,太快了,线程1 也立刻写回去了,就导致 写丢失。
怎么解决? 加锁 或者 用原子类 atomicInteger atomicRefence -
禁止指令重排序
happen-before
image.png
单线程里保证 程序最终执行结果和代码顺序执行的结果一致。
高考写卷子不一定按顺序写 效果高,答题可以先写,先把会做的做了。。。。
案例
结果可能是5也可能是6
双检锁,单例模式,安全性不是百分百
instance = new Singleton() 分为三条指令
1 memory = allocate(); 分配内存
2 instance(memory) 初始化对象
3 instance = memory 设置instance刚刚分配的内存地址,此时intance != null
正常顺序是1,2,3 但是指令重排后可能是 1,3,2 instance != null 但是对象还没有初始化完成!
凭什么atomicInteger 能解决原子性 又不用加锁
CAS, Unsafe类+自旋锁
cas :比较和交换
既保证了一致性 又提高了并发性
syn 一致性保证,并发性下降
cas : 如果这个线程倒霉,循环时间长,cpu开销大
只能保证一个共享变量的原子操作,对于多个共享变量的操作,无法保证操作的原子性,只能加锁
ABA问题是啥? 原子更新引用知道吗?
狸猫换太子? 共享变量是A t1 和 t2 同时拿到A, t2吧A改成B ,然后吧B改成A, t1去执行的时候发现太好了,还是A,我可以改了,但是中间已经被改过又改回来了。
首尾是一样的, 过程不知道被改了多少次了。。
引发的问题:
解决ABA: 带时间戳 或者说版本号,改一次就会变
单链表组成的并发栈问题举例~
cas和 synchroized使用场景
- 资源竞争较少的情况使用synchroized 同步进行线程阻塞和唤醒以及内核的切换会消耗额外的cpu,而CAS基于硬件实现,不需要进入内核,不需要切换线程,操作自旋几率较少,可以获得更好的性能
- 对于资源竞争严重的情况,CAS自旋的概率会比较大,从而浪费更多的CPU资源
- 使用CAS在线程冲突严重的情况下,会大幅降低程序性能,CAS只适用于线程冲突较少的情况下使用
- synchronized在jdk1.6之后,依靠Lock-Free,基本思路是自旋后阻塞,在线程冲突较少的情况下,可以获得和CAS类似的性能,而线程冲突严重的情况下,性能远高于CAS
wait和sleep的区别
sleep()是使线程暂停执行一段时间的方法,是Thread的静态方法, 不会释放锁资源
wait()也是一种使线程暂停执行的方法。是Object的 fangfa 会释放锁资源