详解Java锁对象

news2024/12/26 11:21:41

1、Synchronized

1.1、synchronized特性

1、互斥

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

*进入synchronized修饰的代码块,就相当于加锁。

*退出synchronized修饰的代码块,就相当于解锁。

 一个线程先上了锁,其他线程只能等待这个线程释放。可以理解为“阻塞等待”

注意:

*上一个线程解锁之后,下一个线程并不是立即就可以获取到锁,而是要靠操作系统来“唤醒”,这也就是操作系统线程调度的一部分。

*假设有A,B,C三个线程线程A先获取到锁,然后B尝试获取锁,然后C再尝试获取锁,此时B和C都在阻塞队列中等待,但是A释放锁之后,虽然B比C先来,但B不一定就能拿到锁,而是和C重新竞争。并部遵循先来后到的原则。

2、刷新内存

synchronized工作过程

(1)获得互斥锁

(2)从主内存拷贝变量的最新副本到工作内存

(3)执行代码

(4)将更改后的共享变量的值刷新到主内存

(5)释放互斥锁

所以synchronized可以保证内存可见性

3、可重入

synchronized同步块对同一条线程来说是可重入的,不会出现自己把自己锁死的现象。

锁死现象:

一个线程没有释放锁,然后又开始尝试获取锁

// 第一次加锁, 加锁成功
lock();
// 第二次加锁, 锁已经被占用, 阻塞等待.
lock();

 

 java中的synchronized是可重入锁,所以不存在上述问题。

代码示例:


public class Exe_03 {
    private static int num=50000;
    public static void main(String[] args) throws InterruptedException {
        Counter04 counter1=new Counter04();
        //创建线程完成自增
        Thread thread1=new Thread(()->{
            for (int i = 0; i < num; i++) {
                //自增操作
                counter1.increase();
            }
        });
        Thread thread2=new Thread(()->{
            for (int i = 0; i < num; i++) {
                //自增操作
                counter1.increase1();
            }
        });
        //启动线程
        thread1.start();
        thread2.start();
        //等待线程结束
        thread1.join();
        thread2.join();
        //获取自增后的count值
        System.out.println("count结果="+counter1.count);
    }
}
class Counter04{
    int count=0;
    public synchronized void increase(){
        count++;
    }
    public synchronized void increase1(){
        count++;
    }
}

运行结果:

 在可重入锁的内部,包含了“线程持有者”和“计数器”两个信息。

*如果某个线程加锁的时候,发现锁已经被人占用,但是恰好占用的是自己,那么依然可以获取到锁,并让计数器自增。

*解锁的时候计数器递减为0的时候,才真正的释放锁。(才能被其他线程获取到锁)

1.2、关于synchronized

1、被synchronized修饰的代码,变成串行执行

2、使用多线程的前提是保证结果的正确

3、在多线程修改共享变量的时候才会出现线程安全问题

针对修改操作加锁,缩小锁的范围(锁的粒度)从而提高成序的并发处理能力

4、synchronized不仅可以修饰方法,还可以修饰代码块

 5、用synchronized修饰的代码块所涉及的指令,并不是在CPU一下子就执行完成,而是有可能在执行了一半被CPU调度走了,但是锁并没有释放,别的线程想要获取锁依然需要等待。

6、只给一个线程加锁,也会出现线程安全问题

代码示例:


public class Exe_02 {
    private static int num=50000;

    public static void main(String[] args) throws InterruptedException {
        //实例化对象
        Counter03 counter03=new Counter03();
        //创建线程
        Thread t1=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                //调用加锁方法
                counter03.add();
            }
        });
        Thread t2=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                //调用不加锁方法
                counter03.increase();
            }
        });
        //启动线程
        t1.start();
        t2.start();
        //等待线程执行完成
        t1.join();
        t2.join();
        //打印结果
        System.out.println("累加结果"+counter03.count);
    }
}
class Counter03{
    int count=0;
    public void add(){
        synchronized (this){
            count++;
        }
    }
    public void increase(){
        count++;
    }
}

7、锁对象

 获取锁总结:

1、只有一个线程A要获取锁,那么可以直接获取到锁,没有锁竞争。

2、线程A与线程B共同抢一把锁,那么谁先拿到就先执行谁的逻辑,另外一个线程就阻塞等待,等待持有锁的线程释放锁之后再去抢锁,这时就存在锁竞争。

3、线程A和线程B竞争的不是同一把锁,那么它们没有竞争关系,那么它们分别难道自己的锁,不存在锁竞争关系。

2、锁对象

锁对象本身就是一个对象,无论什么对象都可以成为锁对象(实例对象、类对象)

锁对象记录的是当前获得到锁的线程信息

比如ThreadA拿到了锁,锁信息记录的就是ThreadA的地址。

代码示例:


public class Exe_03 {
    private static int num=50000;
    public static void main(String[] args) throws InterruptedException {
        Counter04 counter1=new Counter04();
        Counter04 counter2=new Counter04();
        //创建线程完成自增
        Thread thread1=new Thread(()->{
            for (int i = 0; i < num; i++) {
                //自增操作
                counter1.increase();
            }
        });
        Thread thread2=new Thread(()->{
            for (int i = 0; i < num; i++) {
                //自增操作
                counter2.increase1();
            }
        });
        //启动线程
        thread1.start();
        thread2.start();
        //等待线程结束
        thread1.join();
        thread2.join();
        //获取自增后的count值
        System.out.println("count结果="+counter2.count);
    }
}
class Counter04{
    int count=0;
    public synchronized void increase(){
        count++;
    }
    public synchronized void increase1(){
        count++;
    }
}

 

 

 ​​​

3、Java Object LayOut工具

3.1、使用Java Object LayOut工具之前要先在pom.xml文件导入maven依赖包

 

代码示例:


import org.openjdk.jol.info.ClassLayout;

public class Demo_405 {
    private int count;
    private long count1 = 200;
    private String hello = "";

    private TestLayout test001 = new TestLayout();

    public static void main(String[] args) {
        Demo_405 obj = new Demo_405();
        // 打印实例布局
        System.out.println(ClassLayout.parseInstance(obj).toPrintable());
        // 调用hashCode后才保存hashCode的值
        obj.hashCode();
        // 观察现象
        System.out.println(ClassLayout.parseInstance(obj).toPrintable());
        System.out.println("=================================");
        // 加锁后观察加锁信息
        synchronized (obj) {
            System.out.println(ClassLayout.parseInstance(obj).toPrintable());
            synchronized (obj) {
                System.out.println(ClassLayout.parseInstance(obj).toPrintable());
            }
            System.out.println(ClassLayout.parseInstance(obj).toPrintable());
        }
        System.out.println("=================================");
        // 强制执行垃圾回收
        System.gc();
        // 观察GC计数
        System.out.println(ClassLayout.parseInstance(obj).toPrintable());
        // 打印类布局
        System.out.println(ClassLayout.parseClass(Demo_405.class).toPrintable());
    }
}

class TestLayout {

}

 

 在多线程环境下进行锁竞争的时候,一个线程抢锁的时候,JVM先看一下对象里有没有锁信息,如果没有,那么就让现在锁竞争的线程获取锁,如果有这个锁信息,那么就阻塞等待锁的释放。

4、锁对象示例

4.1、Locker对象不同代码示例:


public class Exe_06 {
    private static int num = 50000;

    public static void main(String[] args) throws InterruptedException {
        Counter06 counter1=new Counter06();
        Counter06 counter2=new Counter06();
       // counter1.count=1;
        System.out.println(counter1.count);
        System.out.println(counter2.count);
        System.out.println(counter1);
        System.out.println(counter2);
        System.out.println(counter1.locker);
        System.out.println(counter2.locker);
        Thread t1=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                // 这里调用了counter1的自增方法
                counter1.increase();
            }
        });
        Thread t2=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                // 这里调用了counter2的自增方法
                counter2.increase();
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("count结果"+counter1.count);
    }
}
class Counter06{
    public static int count=0;
    //自定义一个锁对象
    Object locker=new Object();
    public void increase(){
        synchronized (locker){
            count++;
        }
    }
}

 4.2、Locker对象相同代码示例:


public class Exe_06 {
    private static int num = 50000;

    public static void main(String[] args) throws InterruptedException {
        Counter06 counter1=new Counter06();
        Counter06 counter2=new Counter06();
       // counter1.count=1;
        System.out.println(counter1.count);
        System.out.println(counter2.count);
        System.out.println(counter1);
        System.out.println(counter2);
        System.out.println(counter1.locker);
        System.out.println(counter2.locker);
        Thread t1=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                // 这里调用了counter1的自增方法
                counter1.increase();
            }
        });
        Thread t2=new Thread(() ->{
            for (int i = 0; i < num; i++) {
                // 这里调用了counter2的自增方法
                counter2.increase();
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("count结果"+counter1.count);
    }
}
class Counter06{
    public static int count=0;
    //自定义一个锁对象
    public static Object locker=new Object();
    public void increase(){
        synchronized (locker){
            count++;
        }
    }
}

 

 4.3、锁静态Locker和this:

4.4、锁类对象:

 4.5、总结:

synchronized的几种用法

1、修饰普通方法,相当于锁实例对象。

2、对代码块加锁,相当于锁当前调用方法的对象(也是实例对象)。

3、对静态方法加锁,相当于锁类对象(静态对象)。

5、valatile关键字

volatile能保证内存可见性,也可以解决有序性问题(禁止指令重排)。

代码在写入volatile修饰的变量的时候,

*改变线程工作内存中volatile变量副本的值。

*将改变后的副本的值从工作内存刷新到主内存。

代码在读取volatile修饰的变量的时候,
*从主内存中读取volatile变量的最新值到工作内存中

*从工作内存中读取volatile变量的副本

观察内存可见性:


import java.util.Scanner;

public class Exe_10 {
    public static class Counter{
        public static volatile int count=0;
    }

    public static void main(String[] args) {
        Thread t1=new Thread(() ->{
            System.out.println(Thread.currentThread().getName()+"线程启动");
            while(Counter.count==0){
                //一直循环
            }
            System.out.println(Thread.currentThread().getName()+"线程退出");
        },"t1");
        //启动线程
        t1.start();
        //确保t1先启动
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        Thread t2=new Thread(() ->{
            System.out.println(Thread.currentThread().getName()+"线程启动");
            Scanner sc=new Scanner(System.in);
            System.out.println("请输入一个非零的值:");
            Counter.count=sc.nextInt();
            System.out.println(Thread.currentThread().getName()+"线程退出");
        },"t2");
        //启动线程
        t2.start();
    }
}

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

10.安装dashboard

有了之前我们部署的 keystone 、 glance 、 nova 、 neutron 服务之后&#xff0c;我们就可以启动云主机了&#xff0c;但是如果只是使用命令来操作OpenStack 的话非常不方便&#xff0c;我们使用OpenStack 搭建云平台就是为了把底层所有资源整合在一起&#xff0c;然后以一种方…

html5前端学习2

一篇思维题题解&#xff1a; 第五周任务 [Cloned] - Virtual Judge (vjudge.net) http://t.csdn.cn/SIHdM 快捷键&#xff1a; CtrlAltDown 向下选取 CtrlAltUp 向上选取&#xff08;会出现多个光标&#xff0c;可以同时输入&#xff09; CtrlEnter …

linux 内核资源配置--cgroups详解以及在docker中的应用

一、Cgroup 概述 1.1、cgroups 是什么 Linux cgroup &#xff08;Control Groups&#xff09;是 Linux 内核提供的一种机制&#xff0c;用于限制进程组使用的资源&#xff08;如 CPU、内存、磁盘 I/O 等&#xff09;。通过将进程组划分为层次结构&#xff0c;并将资源限制应用…

【操作系统】2、进程与线程

【重要考点】 1. 进程与线程 进程与线程的基本概念 进程与线程的状态 转换 ——五态图&#xff0c;七态图 线程的实现&#xff1a; 内核级 用户级 进程与线程的组织和控制 进程间通信IPC&#xff1a; 共享内存、消息传递、管道 2.CPU调度与上下文切换 调度的基本概念 调度的实…

39.SpringCloud—配置管理nacos、远程调用Feign、服务网关Gateway

目录 一、SpringCloud。 &#xff08;1&#xff09;Nacos配置管理。 &#xff08;1.1&#xff09;统一配置管理。 &#xff08;1.2&#xff09;配置热更新&#xff08;自动更新&#xff0c;不需要重启服务&#xff09;。 &#xff08;1.3&#xff09;多环境配置共享。 &a…

【C++篇】初识C++

友情链接&#xff1a;C/C系列系统学习目录 知识点内容正确性以C Primer&#xff08;中文版第五版&#xff09;、C Primer Plus&#xff08;中文版第六版&#xff09;为标准&#xff0c;同时参考其它各类书籍、优质文章等&#xff0c;总结归纳出个人认为较有逻辑的整体框架&…

E. Tracking Segments - 二分+前缀和

分析&#xff1a; 记录所有区间和给定的每一次的询问&#xff0c;二分询问的最小满足条件&#xff0c;可以通过前缀和来计算区间内有几个1。 代码&#xff1a; #include <bits/stdc.h>#define x first #define y secondusing namespace std;typedef long long ll; type…

​面试官疯了:while(true)和for(;;)哪个性能好?

△Hollis, 一个对Coding有着独特追求的人△ 这是Hollis的第 427 篇原创分享 作者 l Hollis 来源 l Hollis&#xff08;ID&#xff1a;hollischuang&#xff09; while(true)和for(;;)都是做无限循环的代码&#xff0c;他俩有啥区别呢&#xff1f; 我承认这个问题有点无聊&#…

四、Docker镜像

学习参考&#xff1a;尚硅谷Docker实战教程、Docker官网、其他优秀博客(参考过的在文章最后列出) 目录 前言一、Docker镜像1.1 概念1.2 UnionFS&#xff08;联合文件系统&#xff09;1.3 Docker镜像加载原理1.4 重点理解 二、docker commit 命令2.1 是什么&#xff1f;2.2 命令…

走进人工智能|机器学习 解码未来的科技革命

前言: 机器学习的发展为我们提供了更智能、高效和便捷的科技产品和服务&#xff0c;可以改善我们的生活和工作方式。 文章目录 序言背景解码未来的科技革命技术支持应用领域程序员如何学总结 序言 机器学习是一种人工智能领域的技术&#xff0c;它让计算机通过数据自动地学习和…

folium离线地图使用

几点说明&#xff1a; 1. 代码亲测有效&#xff08;效果见文末图&#xff09; 2. 离线使用&#xff0c;需要预先下载png格式的离线地图&#xff0c;这里以OpenStreetMap为例。 3. 离线地图下载工具&#xff1a;Offline Map Maker https://www.allmapsoft.com/omm/ 4. foliu…

赋予女性力量:在 Web3 和元宇宙中释放新的机会

Web3 和元宇宙的出现引发了数字领域的革命&#xff0c;为全世界的用户带来了更多可能性。这一转变的其中一个重要方面是赋予女性权力&#xff0c;因为她们更踊跃参与元宇宙活动&#xff0c;并利用 Web3平台挑战传统边界。 The Sandbox 非常自豪能够开放、具包容性地为 womenint…

Swing如何使用?几个操作示例教会你!

以下是四个使用 Swing 组件创建的示例代码&#xff1a; 示例一&#xff1a;JFrame JFrame 是一个顶级容器&#xff0c;用于创建框架窗口。下面是一个简单的 JFrame 示例&#xff1a; import javax.swing.JFrame;/*** author: Ajie* create: 2023/6/21* Description:* FileNa…

OpenStack(1)-创建实例

目录 一、上传镜像 1.1 新建目录 1.2 上传至glance 1.3 查看镜像 二、新建实例 2.1 获取秘钥 2.2 新建实例 2.3 新建实例admin-vm 2.4 获取实例VNC的url 2.5 nova常用命令 一、上传镜像 1.1 新建目录 上传名为cirros-0.3.4-x86_64-disk.img的Linux测试镜像&#xf…

JdbcTemplate 数据访问工具

文章目录 前言JdbcTemplate 数据访问工具1. 概述2. 主要功能3. 示例 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三连啊&#xff0c;写作不易啊^ _ ^。   而且听说点赞的人每天的运气都不会太差&#xff0c;实在白嫖的话&…

新时代的黑科技:就凭这个,还真就非JNPF低代码不可了

随着技术的不断进步&#xff0c;现代企业对软件开发的需求越来越迫切&#xff0c;传统的软件开发方式已无法满足快速迭代和创新的需求。在这种情况下&#xff0c;低代码开发平台应运而生。低代码开发平台通过提供可视化的开发工具和预构建的组件&#xff0c;大大加快了应用程序…

Docker的安装部署以及配置的操作流程(图文)

Docker的安装以及配置流程&#xff08;图文&#xff09; Docker一、配置域名解析二、CentOS Docker 安装1. 查询已安装的docker2. 安装必要的一些系统工具3. 添加软件源&#xff08;阿里云&#xff09;信息4. 更新并安装Docker-CE5. 查看docker 的版本6. 关闭运行的防火墙7. 开…

CMake 工程实践指南专栏介绍

CMake 工程实践指南专栏介绍 大家好,我是 eg,欢迎来到我的 CMake 工程实践指南专栏。本专栏会保持一周一更的节奏,带大家从一个 CMake 小白蜕变成 CMake 熟练使用者。 1. 我与 CMake 的故事 在正式内容开始之前,我想谈谈我和 CMake 的故事。我读书的时候是压根儿不知道 …

造手机、“+ AI”,掉队的蔚来瞎折腾?

对标豪华品牌BBA&#xff0c;自建换电站&#xff0c;蔚来汽车曾凭借先发优势&#xff0c;一度成为造车新势力的领头羊&#xff0c;但最近两个月&#xff0c;蔚来的单月交付量仅有6000多台&#xff0c;渐露掉队之相。 主业低迷&#xff0c;蔚来传出新消息。两天前&#xff0c;工…

c++lambda函数笔记

1、labmda函数用途&#xff1a; 用于简短功能函数的定义&#xff0c;并传递到std算法中。 2、一般函数与lambda函数比较示例 3、如何定义lambda 如下为lambda通用定义式子&#xff1a; [capture] (params) opt->ret{body;}; capture——捕获列表&#xff0c;[]为不捕获变量…