Java编程--synchronized/死锁/可重入锁/内存可见性问题/wait()、notify()

       前言 

        逆水行舟,不进则退!!!     


目录

       线程安全

       synchronized原子锁

       可重入锁(递归锁)

       死锁

       内存可见性问题

       wait()、notify()


       线程安全

        线程安全是指在多线程环境下,程序的行为表现仍然符合我们预期,也就是说,在单线程环境下应该的结果,在多线程环境下也能保证。如果多线程环境下代码运行的结果是不符合我们预期的,即出现数据污染等意外情况,则这个程序就是线程不安全的。

        导致线程不安全的主要因素包括抢占式执行、共享变量等。当存在多个线程并行执行且可能会同时访问和修改同一块内存区域时,如果没有进行适当的同步控制,就可能出现线程安全问题。

        线程安全问题的罪恶之源就是多线程之间的抢占式执行。由于线程的前瞻是执行,导致当前执行到任意一个指令的时候,线程都可能被调度走,cpu 让别的线程来执行。

      


       synchronized原子锁

        synchronized 也叫做 同步机制,要解决线程安全问题,我们就需要将那些有抢占式执行安全隐患的代码原子化,也就是将代码的执行过程变的不可拆分。这样就杜绝了抢占式执行的安全隐患。 

public class ThreadDemo2 {
    //静态成员属性
    static int count = 0;
    /*public static synchronized void counter() {
        count++;
    }*/

    // 静态成员方法 不加锁
    public static void counter() {
        count++;
    }
    
    
    public static void main(String[] args) throws InterruptedException {
        
        Thread t1 = new Thread(() -> {
            for(int i = 0; i < 50000; i++) {
                counter();
            }
        });

        Thread t2 = new Thread(() -> {
            for(int i = 0; i <50000; i++) {
                counter();
            }
        });

        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(count);
    }
}

        上面代码中, 有一个静态成员属性,还有一个静态方法对这个静态成员属性进行自增操作,然后呢,创建了两个线程,同时调用这个静态方法,每个线程调用50000次,按照这样的逻辑来说,最后count的值应该是100000,但是实际运行结果表示,100000只是在概率上可能达到。多次运行结果后,count值的范围出现在50000 到 100000之间。

        要弄清楚上面代码中的线程安全问题,我们首先要清楚一个知识点:在Java中,自增操作看上去是一步执行完毕的,

        实际上分为3个步骤:

                1,先把内存中的值 读取到 CPU 的寄存器中   load 操作

                2,把 CPU 寄存器里的数值进行 +1 运算    add 操作

                3,把得到的结果写回到内存中                  save 操作

        

         

        在抢占式执行的环境下,多线程之间的执行顺序由无数种可能。synchronized 的出现,可以让一次自增操作变得原子化,将自增的这个操作变得不能分割。给自增操作上了锁之后,当线程1在进行自增操作时,若是线程2 也要进行自增操作,那就只能阻塞等待,等待线程1的自增操作执行完毕,释放锁之后,然后线程2 才能进行 自增操作。

        注意:synchronized 只是将执行步骤锁住,并不是说在线程在执行上锁代码时不能被CPU调度,线程是可以被调度走的,若是没有执行完就被调度走,其他阻塞等待的线程 也就只能继续等待。 直到锁被释放

        如果两个线程针对同一个对象进行加锁,就会出现所竞争/ 锁冲突。一个线程能够获取到锁(先到先得),另一个线程则阻塞等待,等待到上一个线程解锁,它才能获取锁成功。

        如果两个线程针对不同对象加锁,此时不会发生锁竞争/ 锁冲突,这俩线程都能获取到各自的锁,不会有阻塞等待了。

        现在,在上述代码中 用synchronized 修饰 counter() 方法,这样就是对counter() 方法上了锁,这时再执行代码,结果就是我们预期的 100000 次了。

public class ThreadDemo2 {
    static int count = 0;
    /*public static synchronized void counter() {
        count++;
    }*/
    public static synchronized void counter() {
        count++;
    }
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            for(int i = 0; i < 50000; i++) {
                counter();
            }
        });

        Thread t2 = new Thread(() -> {
            for(int i = 0; i <50000; i++) {
                counter();
            }
        });

        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println(count);
    }
}


       可重入锁(递归锁)

        一个线程对同一个对象,连续加锁两次,是否会有问题,如果没有问题,就说明该锁是可重入锁;如果有问题,就说明是不可重入锁。

        也就是,一个线程对一段代码加锁了,但是还未释放锁;紧接着获取到了锁。这连着两次加锁如果可以成功,就说明是可重入。

        现在有个问题是: 可重入加锁为什么能成功? 一般来说,一个线程对一个对象进行加锁后,再有线程想要对这个线程进行加锁操作,就会阻塞。然而随着业务的需求,有一种特殊情况还需要考虑,就是第二次加锁的线程和第一次加锁的线程是同一个。这个时候就要考虑开个绿灯行方便。 这个时候会有一个计数器来计算锁的个数,每加锁一次,计数器自增。同样的每解锁一次,计数器自减,直到计数器自减为0后,其他线程才可进入该对象。       也就是说,在对象被上锁的同时,会记录下,当前的锁是被那个线程所持有。

                

        可重入锁的应用场景:

                定时任务:执行定时任务时,如果任务执行时间可能超过下次计划执行时间,可以使用可重入锁来忽略任务的重复触发,确保该任务只有一个线程在执行。


       死锁

        什么是死锁?

                答:死锁指的是两个或两个以上的进程/线程/运算单元 因为互相竞争资源而导致的一种阻塞现象。具体来说,是这些线程互相持有对方所需的资源,导致它们都无法向前推进,若无外力作用,这些进程都将无法推进下去,从而产生永久阻塞的问题。

        死锁产生的四个必要条件如下:

                1. 互斥条件:线程1 拿到了锁,线程2 就得等着。

                2. 请求与保持条件:线程1拿到了锁A,尝试获取锁B,没得逞,然后锁A也不释放。

                3. 不可剥夺条件:线程1拿到了锁之后,只能等线程1主动释放锁,别的线程抢不走。

                4. 循环等待条件:线程1 拿到了锁A ,申请获取锁B,线程2拿到了锁B,申请获取锁A,线程1在等线程2 释放锁B,线程2 在等 线程1 释放锁A。一直等等等。

                这四个条件是死锁发生的必要条件,只要系统发生死锁,这些条件必然成立。

        虽然说死锁产生的原因有四个,而且是缺一不可的,但是呢,我们若想做到预防死锁,其实就只能破坏第四个条件,因为前三个条件都是锁的基本特性,(至少是针对synchronized 这把锁来说,前三点,想动也动不了。)循环等待是这4个条件里,唯一一个和代码结构相关的,也是程序员可以控制的。


       内存可见性问题

        内存可见性问题是指在多线程环境下,当 A线程 正在使用 对象状态 而 B线程 同时修改该对象状态,而B线程修改的 对象状态 对 A线程 不可见。

        要理解这个问题,我们首先要知道CPU缓存的相关知识。今天的CPU主要采用三层缓存:L1、L2是本地核心内缓存,即一个核一个。如果机器是4核,那就有4个L1和4个L2。L3缓存是所有核共享的,无论你的CPU是几核,这个CPU中只有一个L3。

        由于每个线程执行的时候操作的都是同一个CPU的缓存,这就可能导致某个线程修改了对象的状态,但是在其它线程的缓存中,这个对象的值还没有被更新,这就是内存可见性问题。

         内存可见性问题主要是针对多核CPU架构设计的。对于单核CPU,由于同一时间只有一个线程在执行,不存在多线程竞争导致的数据不一致问题,因此单核CPU不会出现内存可见性问题。

        内存可见性问题的出现,部分源自编译器/JVM在多线程环境下的优化。编译器优化的本质是对代码进行智能分析判断,以提高程序运行效率。然而,这种优化有时可能会产生误判,导致多线程环境下的数据不一致问题,也就是内存可见性问题。

        举个例子:

        在这种情况下,我们需要手动干预,防止编译器做出错误的优化。一个常见的解决方法就是在变量前加上volatile关键字,这可以确保修饰的变量在各个线程中的可见性。

         volatile关键字。(volatile意为 :易变的、易失的)  意思就是告诉编译器,这个变量是会改变的,你一定要每次都重新读取这个变量的内存内容。指不定什么时候就改变了,不能随便优化

        此外,内存屏障指令也可以用来强制刷新工作内存中的值,使得所有线程都能看到最新的值。

        值得注意的是,禁用缓存和编译优化虽然可以解决可见性和有序性的问题,但这样会降低程序的运行效率。因此,在实际编程中,我们需要在保证程序运行效率和数据一致性之间找到一个平衡点。

       


       wait()、notify()

        wait() 和 notify() 可以更好的控制多线程之间的执行顺序。 

        多线程最大的问题是,抢占式执行,随机调度。而随机调度 对程序员来说,非常的不友好。所以就想了一些办法,来控制线程之间的执行顺序。虽然线程在内核里的调度是随机的,但是可以通过一些 API 让线程主动阻塞。中东放弃CPU(给别的线程让路)

        举个例子:t1、t2 俩线程,希望 t1 先干活,干的差不多了,再让 t2 来干,就可以让 t2 先wait(阻塞),等着 t1 干的差不多了,通过 notify 通知 t2, 把 t2 唤醒,让 t2 接着干。

        有wait notify , 那为什么还要有 join 呢?

                答:从功能上说,wait 和 notify 比 join 功能更强,覆盖了 join 的功能。 但是呢,前者使用起来要比 join 麻烦不少。

        这里这个异常:

        这里的这个异常要注意一下,多线程中,很多带有阻塞功能的方法都带这个异常,这些方法都是可以被 interrupt 方法通过这个异常给唤醒

        若wait不加任何参数,那就是一个“死等”,一直等待,直到有其他线程唤醒。

wait执行的操作具体如下:

        1,先释放锁;

        2,进行阻塞等待;

        3,收到通知后,重新尝试获取锁,并且在获取锁之后,继续往下执行。

        如果在执行wait的时候,线程并没有锁,那么会报一个 非法的锁状态异常  ;

        例如:上图代码执行结果 如下:

下图代码中就很好的执行了wait命令:

         这里的wait是阻塞了,阻塞在synchronized代码块里,实际上是释放了锁,此时其他的线程是可以获取到Object这个对象的锁的。

notify:notify这个方法必须在获取到锁之后才能生效。 也就是说在java中notify 是必须和 synchronized 来进行搭配使用,

        此处的 notify 通知得和 wait 配对,

        如果wait 使用的对象和notify 使用的对象不同,则此时notify 不会有效果。(notify 只能唤醒在同一个对象上等待的线程。)

        还有一个问题:

这里如果直接这么写,由于线程调度的不确定性,有可能是t2线程的notify 先执行,这样的话 也没有起到任何作用。

        要时刻牢记,线程是抢占式执行的!!!      

        wait 无参数 就是死等

        wait 带参数,就是指定了最大等待时间。

        wait看起来和sleep 有点像:虽然都是能指定等待时间,虽然也都能被提前唤醒(wait使用notify唤醒,sleep使用interrupt唤醒)。但是表示的含义截然不同。notify唤醒wait 是不会有任何异常的。(正常的业务逻辑)   interrupt唤醒sleep则是出异常了(表示一个出问题了的逻辑

        

        如果有多个线程在等待object对象,此时有一个线程在执行 object.notify(),这时是随机唤醒一个等待的线程,(不知道具体是哪个)    notifyAll是唤醒所有wait object的线程。


        我是专注学习的章鱼哥~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/133584.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

WebSphere Liberty 8.5.5.9 (二)

WebSphere Liberty 8.5.5.9 &#xff08;二&#xff09; encode and decode Pre WebSphere Liberty 8.5.5.9 xor and AES 提取 D:\wlp-webProfile7-java8-8.5.5.9\wlp\lib 下必要加解密包 com.ibm.ws.crypto.certificateutil_1.0.12.jar com.ibm.ws.crypto.passwordutil_1…

C语言 每日一题 PTA 11.7 day13

1.求e的近似值 自然常数 e 可以用级数 1 1 / 1! 1 / 2! ⋯ 1 / n! ⋯ 来近似计算。 本题要求对给定的非负整数 n&#xff0c;求该级数的前 n 1 项和。 代码实现 #include<stdio.h> void main() {int a, i, j; double b 1; double c 1;printf("请输入一个数\n…

华为ensp:静态默认路由

静态路由 到r2 上的系统视图模式 下一跳为1.1.1.2 ip route-static 192.168.2.0 255.255.255.0 1.1.1.2 如果找2网段下一跳为1.1.1.2接口 默认路由 到r3上做的是默认路由 ip route-static 0.0.0.0 0 1.1.1.1 所有的流量去找1.1.1.1 查看效果 只要做完完整的路由就可…

思维模型 超限效应

本系列文章 主要是 分享 思维模型&#xff0c;涉及各个领域&#xff0c;重在提升认知。物极必反。 1 超限效应的应用 1.1 教育中的超限效应 一位老师在课堂上批评了一位学生&#xff0c;这位学生可能会因为老师的批评而感到沮丧和失落。如果老师在接下来的课程中继续批评这位…

【Armstrong公理】【求闭包和候选码】【判断范式】

1. Armstrong公理 2.求闭包和候选码 3.判断范式

Scrum敏捷开发全流程,3款必备的项目管理工具!

​Scrum是一种敏捷方法&#xff0c;致力于帮助团队高效地协作和完成复杂的项目。它强调迭代和快速迭代、自组织、快速响应变化等原则&#xff0c;使得项目开发变得更加灵活和高效。 在Scrum敏捷开发过程中&#xff0c;项目管理工具是必不可少的。下面介绍3款常用的敏捷开发工具…

EDA实验----四选一多路选择器设计(QuartusII)

目录 一&#xff0e;实验目的 二&#xff0e;实验仪器设备 三&#xff0e;实验原理&#xff1a; 四&#xff0e;实验要求 五&#xff0e;实验内容及步骤 1.实验内容 2.实验步骤 六&#xff0e;实验报告 七.实验过程 1.创建Verilog文件&#xff0c;写代码 2.波形仿真 …

C语言 每日一题 PTA 11.8 day14

1.矩阵A乘以B 给定两个矩阵A和B&#xff0c;要求你计算它们的乘积矩阵AB。需要注意的是&#xff0c;只有规模匹配的矩阵才可以相乘。 即若A有Ra​行、Ca列&#xff0c;B有Rb行、Cb列&#xff0c;则只有Ca与Rb​相等时&#xff0c;两个矩阵才能相乘。 输入格式&#xff1a; 输入…

【网络编程】网络层——IP协议

文章目录 基本概念路径选择主机和路由器 IP协议格式分片与组装网段划分IP地址的数量限制私网IP地址和公网IP地址深入认识局域网路由 基本概念 TCP作为传输层控制协议&#xff0c;其保证的是数据传输的可靠性和传输效率&#xff0c;但TCP提供的仅仅是数据传输的策略&#xff0c…

selenium+python做web端自动化测试框架实战

最近受到万点暴击&#xff0c;由于公司业务出现问题&#xff0c;工作任务没那么繁重&#xff0c;有时间摸索seleniumpython自动化测试&#xff0c;结合网上查到的资料自己编写出适合web自动化测试的框架&#xff0c;由于本人也是刚刚开始学习python&#xff0c;这套自动化框架目…

开发ios电脑app的费用受到哪方面的影响?

开发iOS电脑应用程序的费用受到多方面的影响&#xff0c;包括市场需求、功能复杂度、设计要求、开发人员经验、市场竞争以及后期维护等因素&#xff0c;下面我们将详细介绍这些影响因素&#xff0c;帮助您更好地了解开发iOS应用程序的费用构成。 一、市场需求 市场需求是影响…

puzzle(1612)拼单词、wordlegame

目录 拼单词 wordlegame 拼单词 在线play 找出尽可能多的单词。 如果相邻的话&#xff08;在任何方向上&#xff09;&#xff0c;你可以拖拽鼠标从一个字母&#xff08;方格&#xff09;到另一个字母&#xff08;方格&#xff09;。在一个单词中&#xff0c;你不能多次使用…

计算机网络技术

深入浅出计算机网络 微课视频_哔哩哔哩_bilibili 第一章概述 1.1 信息时代的计算机网络 1. 计算机网络各类应用 2. 计算机网络带来的负面问题 3. 我国互联网发展情况 1.2 因特网概述 1. 网络、互连网&#xff08;互联网&#xff09;与因特网的区别与关系 如图所示&#xff0…

C语言——个位数为 6 且能被 3 整除但不能被 5 整除的三位自然数共有多少个,分别是哪些?

#define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h> int main() {int i,j0;for(i100;i<1000;i) {if(i%106&&i%30&&i%5!0){printf("%6d",i); j;}}printf("\n一共%d个\n",j);return 0; } %6d起到美化输出格式的作用&#xff…

[工业自动化-11]:西门子S7-15xxx编程 - PLC从站 - 分布式IO从站/从机

目录 一、什么是以分布式IO从站/从机 二、分布式IO从站的意义 三、ET200分布式从站系列 一、什么是以分布式IO从站/从机 在工业自动化领域中&#xff0c;分布式 IO 系统是目前应用最为广泛的一种 I/O 系统&#xff0c;其中分布式 IO 从站是一个重要的组成部分。 分布式 IO …

centos7安装linux版本的mysql

1.下载linux版本的mysql 进入mysql官网&#xff0c;点击社区版本下载&#xff1a; https://dev.mysql.com/downloads/mysql/ 选择版本&#xff0c;可以跟着我下面这个图进行选择&#xff0c;选择红帽版本的既可&#xff0c;都是linux版本的。 2.上传解压linux版本的mysql安装包…

悟空crm二次开发 增加客户保护功能 (很久没有消息,但是有觉得有机会的客户)就进入了保护转态

需求&#xff1a;客户信息录入不限数量&#xff0c;但是录入的信息1个月内只有自己和部门领导能看到&#xff0c;如果1个月内未成交或者未转移至自己的客保 则掉入公海所有人可见&#xff0c;这里所说的客保就是现在系统自带的客保 1、需求思维导图 2、新增保护按钮 3、点击该…

transformers安装避坑

1.4 下载rust编辑器 看到这里你肯定会疑惑了&#xff0c;我们不是要用python的吗&#xff1f; 这个我也不知道&#xff0c;你下了就对了&#xff0c;不然后面的transformers无法安装 因为是windows到官网选择推荐的下载方式https://www.rust-lang.org/tools/install。 执行文…

Netty入门指南之NIO Selector监管

作者简介&#xff1a;☕️大家好&#xff0c;我是Aomsir&#xff0c;一个爱折腾的开发者&#xff01; 个人主页&#xff1a;Aomsir_Spring5应用专栏,Netty应用专栏,RPC应用专栏-CSDN博客 当前专栏&#xff1a;Netty应用专栏_Aomsir的博客-CSDN博客 文章目录 参考文献前言问题解…

skynet学习笔记03— 服务

01、API newservice(name, ...)&#xff1a; 阻塞的形势启动一个名为 name 的新服务&#xff0c;待start函数执行完后会返回这个服务的地址。uniqueservice(name, ...)&#xff1a;针对于当前节点&#xff0c;启动一个唯一服务&#xff08;相当于单例&#xff09;&#xff0c;…