Volatile关键字

1.保证线程可见性
    不可见:两个线程 当一个线程操作这个值,其他线程不知道,用的还是原来的值
    底层用了 cpu的缓存一致性协议
2.禁止指令重排序
    不是禁止cpu的指令排序

单例:就是保证在JVM内存里,永远只有某一个类的一个实例
双重检查  , 加Volatile重点,不加Volatile 问题就会出现在指令重排序上,一般并发超高的时候才会出现
锁细化:加锁给少量代码,不用锁住一个方法

如果两个线程同时都判断 instance 为null了 ,一个线程加锁完释放掉后,另一个线程又会重新初始化一次变量了
所以要用双重检查单例:
public class Mgr01(){
    private static volatile Mgr01 INSTANCE;

    privateMgr01(){

    }
    
    public static Mgr01 getInstance(){
        //双重检查
        if(INSTANCE == null){
            synchronized(Mgr01.class){
                if(INSTANCE == null){
                    try{
                        Thread.sleep(1);
                    }catch(Exception e){
                        e.printStackTrace();
                    }
                    INSTANCE = new Mgr01();    
                }
            }
        }
        return INSTANCE;
    }

    public static void main(String []args){
        for(int i = 0;i<100;i++){
            new Thread(()->{
                System.out.println(Mgr01.getInstance().hashCode());        
            }).start();
        }
    }
}

INSTANCE = new Mgr01();  :  new一个对象,根据编译器编译完,这句指令其实是分为三步:
第一步 是给这个对象申请内存  默认值 int类型 是0
第二步 是初始化成员变量
第三步 把内存这块的内容赋值给INSTANCE    instance是在栈内存里

Volatile不保证原子性:
Volatile 并不能保证多个线程共同修改running 变量时所带来的不一致问题  也就是说Volatile不能替代synchronized
count 保证可见性的操作,但是count++ ,它本身并不是一个原子性的操作 

new Thread(t::m,"t1").start();
t::m   : lambda表达式  new Thread(new Runnable(run(){m()}).start();
有一个方法名叫m

    

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值