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


      什么是线程安全的类?

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

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

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

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

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

      原子性

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

      当需要同步时,可能我们首先想到的就是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原则推到出来,就不能保证有序性,虚拟机可随意的对他们进行重排序。

      

  • 相关阅读:
    [Codeforces721E]Road to Home
    [Codeforces513E2]Subarray Cuts
    [CodeForces332E]Binary Key
    [HDU4585]Shaolin
    [HDU3726]Graph and Queries
    [BZOJ3224]普通平衡树
    [BZOJ3173]最长上升子序列
    [POJ2985]The k-th Largest Group
    PHP一句话
    体验VIP版本灰鸽子,哈哈,拿到了老师的病毒教程
  • 原文地址:https://www.cnblogs.com/sbrn/p/9001422.html
Copyright © 2020-2023  润新知