volatile是java提供的轻量级同步机制有以下几个特征:
保证可见性通过给变量声明volatile属性,当变量的值在共享内存发生改变之后马上将值同步到线程的工作内存。不保证原子性
a.在变量的值在线程的工作内存中被修改后同步到共享内存是发生了写覆盖,把其线程的值覆盖了。
如何解决?
使用JUC中的Atomic类来修改变量的值 为什么要在单例对象前声明volatile
单例对象的创建分为三个过程
1.为对象分配内存空间
2.初始化对象
3.将分配好的内存空间指向对象
由于2,3不存在数据依赖关系,编译后指令重排的代码可能会发生将内存空间指向空的对对象,这样导致初始化的对象没有分配内存空间。其他线程拿到单例对象使用发生空指针异常。
ublic class SingletonDemo {private volatile static SingletonDemo instance = null;public SingletonDemo() {System.out.println(Thread.currentThread().getName()+"我是构造方法");}public static SingletonDemo getInstance(){if (instance==null){synchronized (SingletonDemo.class){if (instance==null){instance =new SingletonDemo();}}}return instance;}public static void main(String[] args) {for (int i = 0; i < 10; i++) {new Thread(()->{SingletonDemo.getInstance();},String.valueOf(i)).start();}}}