线程的状态及线程安全

线程的状态

线程的所有状态:

线程的状态是一个枚举类型:Thread.State

线程通过状态标识当前线程所处的一个情况

1.JVM内部管理线程

2.提供给代码的编写者观察程序的运行状态

public class ThreadState {
    public static void main(String[] args) {
        for(Thread.State state:Thread.State.values()){
            System.out.println(state);
        }
    }
}

线程的状态转移

isAlive()  判断线程是否活着   //线程只要不处于不是new和terminated就是huozhe

join()  //相当于一直在判断线程是否活着的死循环

yield()  //可以理解位一个雷锋同志,当线程抢到CPU时,主动让出CPU资源,重新回到排队队伍中去

线程安全

线程安全:程序按照预期正确工作

线程不安全:程序没有按照预期工作

要保证线程安全是一件很难的事情(我们的目标是保证线程安全的时候,尽可能追求效率)

出现线程不安全问题的原因:

1.共享资源/共享变量/共享数据(栈里的数据是线程私有的【每个线程都有自己独立的调用栈:局部变量+形参】,堆【除了栈以外】是大家共享的)—— 没有共享,就没有线程不安全问题

PC/栈(java栈+本地栈)—— 私有的  即   局部变量/形参 —— 私有的

堆/方法区/常量池 —— 共享的              即,   对象/类的信息(类的对象)/属性/静态属性 —— 共享的

由此可知:私有数据不需要考虑线程安全问题,共享数据才是导致线程不安全

2.线程之间会因为调度原因,穿插着执行

3.代码的原子性/内存的可见性/代码重排序

代码原子性:一段代码在运行期间是不可分割的(一句java代码不一定是原子的,一条指令也不一定是原子的)

例:n++;n--;(1.先取n的值,2把n的值+1或-1,3.把n+1或n-1的值放回到寄存器中)

不保证原子性会给多线程带来的文题:如果一个线程正在对一个变量操作,中途其他线程进来了,这个操作就被打断了,结果就是错误的

为了解决这个问题,可以给代码加锁

内存的可见性:

计算机的存储三角形:

为了提高效率,JVM在执行过程中,会尽可能的将数据工作内存中执行,但这样会造成一个问题,共享在多线程之间不能及时看到改变,这个就是可见性问题。

代码重排序:

在单线程情况下,JVM/CPU指令集/运行时JIT会对代码进行适度优化

但是在多线程情况下代码就有可能会出错

如何保证线程安全:

1.如果可以设计出不需要共享的数据,天生安全

2.如果非要进行数据的共享,尽可能保证数据的可读性——不可变对象

3.利用各种机制保证以上三点:原子性,可见性,重排序

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值