并发编程(六):线程安全性

时间:2021-01-23 17:56:17

  什么是线程安全的类?

  当多个线程访问某个类时,不管运行时环境采用何种调度方式或者这些进程将如何交替执行,并且在主调代码中不需要额外同步或协同,这个类都能表现出正确的行为,那么就称这个类是线程安全的

 

  线程安全性包含哪些特性?

  原子性:提供了互斥访问,同一时刻只能有一个线程对它进行操作

  可见性:一个线程对主内存的修改可以及时被其他线程观察到

  有序性:一个线程观察其他线程的指令执行顺序,由于指令重排序的存在,该观察结果一般杂乱无序。

 

  原子性

  保证原子性的手段通常有以下几种

并发编程(六):线程安全性

  当需要同步时,可能我们首先想到的就是synchronized,但是在竞争特别激烈时,synchronized可能并不是很好的选择,synchronized的使用方式大概有以下几种

并发编程(六):线程安全性

  关于atomic,我们在前面的博客中已有实例应用,关于Lock的使用我们会在后面的博客中做专门的讲解,下面我们简单看一下synchronized的demo

  synchronized-demo1(修饰方法)

@Slf4j
public class SynchronizedExample1 {

    //修饰一个代码块
    public void test1(int j) {
        synchronized (this) {
            for (int i = 0; i < 10; i++) {
                log.info("test1-{}-{}", j,i);
            }
        }
    }

    //修饰一个方法
    public synchronized void test2(int j) {
        for (int i = 0; i < 10; i++) {
            log.info("test2-{}-{}", j,i);
        }
    }

    public static void main(String[] args) {
        SynchronizedExample1 example1 = new SynchronizedExample1();
        SynchronizedExample1 example2 = new SynchronizedExample1();
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.execute(()->{
            example1.test2(1);
        });

        executorService.execute(()->{
            example2.test2(2);
        });

    }

}

 

  synchronized-demo2(修饰类)

@Slf4j
public class SynchronizedExample2 {

    //修饰一个类
    public static void test1(int j) {
        synchronized (SynchronizedExample2.class) {
            for (int i = 0; i < 10; i++) {
                log.info("test1-{}-{}", j,i);
            }
        }
    }

    //修饰一个静态方法
    public static synchronized void test2(int j) {
        for (int i = 0; i < 10; i++) {
            log.info("test2-{}-{}", j,i);
        }
    }

    public static void main(String[] args) {
        SynchronizedExample2 example1 = new SynchronizedExample2();
        SynchronizedExample2 example2 = new SynchronizedExample2();
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.execute(()->{
            example1.test1(1);
        });

        executorService.execute(()->{
            example2.test1(2);
        });

    }

}

 

  可见性

  导致共享变量在线程间不可见的原因

并发编程(六):线程安全性

  通常我们有两种方式来保证可见性:

  a、synchronized

  java内存模型中关于synchronized有两条规定

  1、线程解锁前,必须把共享变量的最新值刷新到主内存

  2、线程加锁时将清空工作内存*享变量的值,从而使用共享变量时需要从主内存中重新读取最新的值(加锁与解锁必须是同一把锁)

 

  b、volatile

  volatile通过加入内存屏障和禁止重排序优化来实现可见性

  a、对volatile变量写操作时,会在写操作后加一条store屏障指令,将本地内存中的共享变量值刷新到主内存

  b、对volatile变量读操作时,会在读操作前加入一条load屏障指令,从主内存读取共享变量

 

  volatile写:

并发编程(六):线程安全性

 

  volatile读:

并发编程(六):线程安全性

 

  有序性

  java内存模型中,允许编译器和处理器对指令进行重排序,重排序过程不会影响到单线程程序的执行,却会影响到多线程并发执行的正确性,有序性可依靠volatile、synchronized、Lock来保证。

  提起有序性,我们可以想到JMM(java内存模型)中一个非常重要的原则-happens-before原则:

  a、程序次序规则:一个线程内,按照代码顺序,书写在前面的操作先行发生于书写在后面的操作

  b、锁定操作:一个unLock操作先行发生于后面对同一个锁的lock操作

  c、volatile变量规则:对一个变量的写操作先行发生于后面这个变量的读操作

  d、传递规则:如果操作A先行发生于操作B,而操作B又先行发生于操作C,则可以得出操作A先行发生于操作C

  e、线程启动规则:Thread对象的start()方法先行发生于此线程的每一个动作

  f、线程中断规则:对线程interrupt()方法的调用先行发生于被中断线程的代码检测到中断事件的发生

  g、线程终结规则:线程中所有操作都先行发生于线程的终止检测,可有通过Thread.join()方法结束、Thread.isAlive()的返回值手段检测到线程已经终止执行

  h、对象终结规则:一个对象的初始化完成先行发生于他的finalize()(GC在回收对象之前调用该方法)方法的开始

 

  如果两个操作的执行次序,无法从happends-before原则推到出来,就不能保证有序性,虚拟机可随意的对他们进行重排序。