四种方式

start0() native方法 在JVM源码,java其实是应用层,线程并不是java产生的,而是底层操作系统支持线程,jvm调用操作系统去创建线程,针对不同平台做了封装,在各个操作系统都能创建线程,启动了线程之后会去回调run方法

image.png
创建
启动,设置状态为runnabale

怎么终止线程

一定不是stop! 这是个过期的方法,挂起和恢复也不推荐使用,
相当于Kill-9 虽然能完成效果,不安全,不优雅
无法知道被关闭的线程的状态,运行到一半 被关闭了。。

Thread.interrupt() 终止才是对的

六种状态

线程池原理

线程工具类四种方法

为何不推荐使用现成的方法

自己创建线程 七大参数含义

阻塞队列

JMM内存模型

jmm

jmm1

硬盘》内存》cpu
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 :比较和交换


cas概述
image.png

image.png

image.png

image.png

既保证了一致性 又提高了并发性
syn 一致性保证,并发性下降
cas : 如果这个线程倒霉,循环时间长,cpu开销大
   只能保证一个共享变量的原子操作,对于多个共享变量的操作,无法保证操作的原子性,只能加锁

ABA问题是啥? 原子更新引用知道吗?
狸猫换太子? 共享变量是A t1 和 t2 同时拿到A, t2吧A改成B ,然后吧B改成A, t1去执行的时候发现太好了,还是A,我可以改了,但是中间已经被改过又改回来了。
首尾是一样的, 过程不知道被改了多少次了。。


image.png

ABA代码演示

引发的问题:

解决ABA: 带时间戳 或者说版本号,改一次就会变
单链表组成的并发栈问题举例~

cas和 synchroized使用场景

  • 资源竞争较少的情况使用synchroized 同步进行线程阻塞和唤醒以及内核的切换会消耗额外的cpu,而CAS基于硬件实现,不需要进入内核,不需要切换线程,操作自旋几率较少,可以获得更好的性能
  • 对于资源竞争严重的情况,CAS自旋的概率会比较大,从而浪费更多的CPU资源
  • 使用CAS在线程冲突严重的情况下,会大幅降低程序性能,CAS只适用于线程冲突较少的情况下使用
  • synchronized在jdk1.6之后,依靠Lock-Free,基本思路是自旋后阻塞,在线程冲突较少的情况下,可以获得和CAS类似的性能,而线程冲突严重的情况下,性能远高于CAS

wait和sleep的区别

sleep()是使线程暂停执行一段时间的方法,是Thread的静态方法, 不会释放锁资源
wait()也是一种使线程暂停执行的方法。是Object的 fangfa 会释放锁资源

拓展链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容