线程安全问题(面试常考)

目录

🍊一. 观察多线程下n++和n--操作

🍉二. 线程安全概念的引入

🫐三. 线程不安全的原因

🌴1. 原子性

🌾2. 可见性

🌵3. 有序性

4. 线程不安全的原因总结 

🍒四. 解决线程不安全问题

🌿1. synchronized关键字

🍂1.1 语法格式

🍂1.2 sychronized的作用

🍂1.3 对n++,n--代码进行修改 

🌴2. volatile关键字

🌳3. Lock(Java api提供的一个锁,后续在锁策略中介绍) 


🍊一. 观察多线程下n++和n--操作

我们目前所知当一个变量n==0,n++了1000次并且 n--了1000次,我们的预期结果为0,但是当两个线程分别执行++和--操作时最后的结果是否为0呢?

看这样一段代码:

public class ThreadSafe {
    private static int n = 0;

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                        n++;
                }
            }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                        n--;
                }
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(n);
    }
}

看一下分别运行3次的结果:

从结果上看都没有达到我们的预期结果,因为两个线程同时操作一个共享变量时,这其中就涉及到线程安全问题 

🍉二. 线程安全概念的引入

我们所知单线程下n++和n--同时执行1000次时结果为0,多线程下大部分不为0,所以我们简单定义为在多线程下和单线程下执行相同的操作结果相同时为线程安全

对于多个线程,操作同一个共享数据(堆里边的对象,方法区中的数据,如静态变量):

如果都是读操作,也就是不修改值,这时不存在安全问题
如果至少存在写操作时,就会存在线程安全问题

🫐三. 线程不安全的原因

🌴1. 原子性

一组操作(一行或多行代码)是不可拆分的最小执行单位,就表示这组操作是具有原子性

多个线程多次的并发并行的对一个共享变量操作时,该操作就不具有原子性

看这样一个例子,如下图:

这最终导致的结果是一张票被售卖了两次,这样就具有很大的风险性

注意:我们在写的一行Java代码可能不是原子性的,因为它编译成字节码,或者由JVM把字节码翻译为机器码后就不是一行,也就是多条执行操作

典型的n++,n--操作:

经过一次n++,n--操作后发现结果不为-1,原因是因为一次++或者--操作是分三步执行:

🍁从内存把数据读到CPU
🍁对数据进行更新操作
🍁再把更新后的操作写入内存

🌾2. 可见性

多个线程工作的时候都是在自己的工作内存中(CPU寄存器)来执行操作的,线程之间是不可见

1. 线程之间的共享变量存在主内存
2. 每一个线程都有自己的工作内存
3. 线程读取共享变量时,先把变量从主存拷贝到工作内存(寄存器),再从工作内存(寄存)读取数据
4. 线程修改共享变量时,先修改工作内存中的变量值,再同步到主内存

举例子说明上述问题: 

public class Demo {

    private static int flag = 0;
    public static void main(String[] args) {
        Thread t = new Thread(() -> {
            while(flag == 0){

            }
            System.out.println("t线程执行完毕");
        });
        t.start();

        Scanner sc = new Scanner(System.in);
        flag = sc.nextInt();
        System.out.println("main线程执行完毕");
    }
}

 结果:flag的值已经不为0,但是t线程还没执行结束,因为t线程读flag读的是寄存器中的0

🍬为什么要保证可见性?

把保证每次读取变量的值时都从主存获取最新的值

🌵3. 有序性

🍬了解重排序:

JVM翻译字节码指令,CPU执行机器码指令,都可能发生重排序来优化执行效率

比如有这样三步操作:(1) 去前台取U盘 (2) 去教室写作业 (3) 去前台取快递
JVM会对指令优化,也就是重排序,新的顺序为(1)(3)(2),这样来提高效率

4. 线程不安全的原因总结 

1. 线程是抢占式的执行,线程间的调度充满了随机性
2. 多个线程对同一个变量进行修改操作
3. 对变量的操作不是原子性的
4. 内存可见性导致的线程安全
5. 指令重排序也会影响线程安全

🍒四. 解决线程不安全问题

🌿1. synchronized关键字

🍂1.1 语法格式

1. 修饰普通方法,也叫同步实例方法

    public synchronized void doSomething(){
        //...
    }

等同于 

    public void doSomething(){
        synchronized (this) {
            //...
        }
    }

2. 修饰静态方法,也叫静态同步方法

    public static synchronized void doSomething(){
        //...
    }

等同于

    public static void doSomething(){
        synchronized (A.class) {
            //...
        }
    }

3. 修饰代码块

需要显示指定对哪个对象加锁(Java中任意对象都可以作为锁对象) 

    synchronized (对象) {
        //...
    }

🍂1.2 sychronized的作用

sychronized是基于对象头加锁的,特别注意:不是对代码加锁,所说的加锁操作就是给这个对象的对象头里设置了一个标志位

一个对象在同一时间只能有一个线程获取到该对象的锁
sychronized保证了原子性,可见性,有序性(这里的有序不是指指令重排序,而是具有相同锁的代码块按照获取锁的顺序执行)

1. 互斥性

synchronized 会起到互斥效果, 某个线程执行到某个对象的 synchronized 中时, 其他线程如果也执行到同一个对象 synchronized 就会阻塞等待

进入 synchronized 修饰的代码块, 相当于 加锁
退出 synchronized 修饰的代码块, 相当于 解锁

看下图理解加锁过程:

阻塞等待:

针对每一把锁, 操作系统内部都维护了一个等待队列. 当这个锁被某个线程占有的时候,其他线程尝试进行加锁, 就加不上了,就会阻塞等待, 一直等到之前的线程解锁之后, 由操作系统唤醒一个新的线程, 再来获取到这个锁

2. 刷新主存

synchronized的工作过程:

🍃获得互斥锁
🍃从主存拷贝最新的变量到工作内存
🍃对变量执行操作
🍃将修改后的共享变量的值刷新到主存
🍃释放互斥锁

3. 可重入性

synchronized是可重入锁
同一个线程可以多次申请成功一个对象锁

可重入锁内部会记录当前的锁被哪个线程占用,同时也会记录一个加“锁次数”,对于第一次加锁,记录当前申请锁的线程并且次数加一,但是后续该线程继续申请加锁的时候,并不会真正加锁,而是将记录的“加锁次数加1”,后续释放锁的时候,次数减1,直到次数减为0才是真的释放锁

可重入锁的意义就是降低程序员负担(使用成本来提高开发效率),代价就是程序的开销增大(维护锁属于哪个线程,并且加减计数,降低了运行效率) 

如下情形:

🍂1.3 对n++,n--代码进行修改 

public class ThreadSafe {
    private static int n = 0;

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                    synchronized (ThreadSafe.class) {
                        n++;
                    }
                }
            }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                    synchronized (ThreadSafe.class){
                        n--;
                    }
                }
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(n);
    }
}

 结果:结果为我们预期结果,说明是线程安全的

🌴2. volatile关键字

volatile是用来修饰变量的,它的作用是保证可见性,有序性
注意:不能保证原子性,对n++,n--来说,用volatile修饰n也是线程不安全的

· 代码在写入 volatile 修饰的变量的时候,改变线程工作内存中volatile变量副本的值将改变后的副本的值从工作内存刷新到主内存
· 代码在读取 volatile 修饰的变量的时候,从主内存中读取volatile变量的最新值到线程的工作内存中
从工作内存中读取volatile变量的副本

使用场景:

读操作:读操作本身就是原子性,所以使用volatile就是线程安全的
写操作:赋值操作是一个常量值(写到主存),也保证了线程安全

用volatile修饰变量n看是否线程安全:

public class ThreadSafe {
    private static volatile int n = 0;

    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                        n++;
                }
            }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i = 0;i < 1000;i++){
                        n--;
                }
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(n);
    }
}

结果:也是不是线程安全的


 

🌳3. Lock(Java api提供的一个锁,后续在锁策略中介绍) 

  • 57
    点赞
  • 221
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 26
    评论
常考面试题 1.讲下servlet的执行流程。 Servlet的执行流程也就是servlet的生命周期,当服务器启动的时候生命周期开始,然后通过init()《启动顺序根据web.xml里的startup-on-load来确定加载顺序》 方法初始化servlet,再根据不同请求调用doGet或doPost方法,最后再通过destroy()方法进行销毁。 2.Get和Post的区别 在页面提交数据时,get的数据大小有限制4k,post没有限制,get请求提交的数据会在地址栏显示,post不显示,所以post比get安全. 3.有三台服务器,如果在一台服务器上登陆了这个用户,其他两台就不能再登陆这个用户,使用session共享,你是怎么做的。 把所有的session的数据保存到Mysql服务器上,所有Web服务器都来这台Mysql服务器来获取Session数据。 4.写一个自定义标签要继承什么类 SimpleTagSupport,一般调用doTag方法 或者实现SimpleTag接口 5.Jsp如何处理json 在 jsp 中处理 JSON,通常需要配套使用 JQuery 控件,并且导入一些 Common jar 包。 使用 JQuery 控件是因为它能有效的解析并且展示 JSON 数据, 导入Common 则是因为 Java 中的对象并不是纯粹的数据,需要通过这些 Jar 包的处理使之转化成真实数据。 6.如何处理Servlet的线程不安全问题 1.最简单的就是不使用字段变量, 2.使用final修饰变量, 3.线程安全就是多线程操作同一个对象不会有问题,线程同步一般来保护线程安全, 所以可以在Servlet的线程里面加上同步方法或同步块。 (Synchronized)可以保证在同一时间只有一个线程访问,(使用同步块会导致性能变差,最好不去使用实例变量) 7.Jsp的重定向和转发的流程有什么区别 重定向是客户端行为,转发是服务器端行为 重定向时服务器产生两次请求,转发产生一次请求,重定向时可以转发到项目以外的任何网址,转发只能在当前项目里转发 重定向会导致request对象信息丢失。转发则不会 转发的url不会变,request.getRequestDispatch().forward() 重定向的url会改变,response.getRedirect().sendRedirect();

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

X_H学Java

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值