类锁和实例对象锁你分清了吗?

2023-11-06 02:01
文章标签 分清 对象 实例 类锁

本文主要是介绍类锁和实例对象锁你分清了吗?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

系列文章目录

文章目录

  • 系列文章目录
  • 前言
  • 一、什么是锁竞争?
  • 二、什么是类锁?什么是实例对象锁?
  • 三、给类对象加锁不是锁住了整个类
  • 四、总结


前言

java选手们应该都对锁不陌生,加锁了就是为保证操作语句的原子性,如果你是刚学并发编程,是否傻傻分不清楚对象锁和类锁呢?别怕!!!你看到了我的这篇文章就能帮你解决这个困惑~~


一、什么是锁竞争?

当我们使用synchronized个一个对象加上了锁,多个线程尝试在自己的内存空间上拿到这个加了锁的对象时,此时就会发生锁竞争,在竞争的瞬间只有一个线程可以拿到这个加了锁的对象,此时线程就是安全的。
举个例子:

假设你寝室里的卫生间只有一个马桶,某天晚上,你室友们同时都想去上厕所,那么你们就是要去抢这个厕所。
在这里你和你的室友就是线程;
厕所里的马桶就是对象;
厕所门上的锁就是synchronized;

二、什么是类锁?什么是实例对象锁?

类锁就是对类的成员或者方法或者类对象加锁,类锁本质就是对类对象加锁。

什么是类对象?
类对象就是.class对象,类对象详细的记录了程序员在定义这个类时全部的信息,比如:属性、方法等
你可以看到类对象是.class,类对象来源与.class文件,.class文件是由javac编译器根据.java源文件编译出来的,你可以理解成源文件只有一份,所以类对象也只有一份

类锁:
给static修饰的属性或者方法或者直接在synchronized(类.class)都是在给类加锁
实例对象锁:
给非static修饰的属性或者方法加锁

//给count加2000次
class A{static int count;//对静态方法加锁,就是对类对象加锁static synchronized void fun1(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是类方法"+count);}//对实例方法加锁,就是对实例对象加锁synchronized void  fun2(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是实例方法"+count);}
}

三、给类对象加锁不是锁住了整个类

是否线程安全,就看两个线程是否是针尝试获取到同一个加了锁的对象。就算里给类对象加了锁,也不是意味着一个线程拿到锁了,其他线程只能阻塞等待,如果其他线程本来就没有要去获取到这把类锁,而是去获取到实例对象的锁,那么这里就不存在多个线程竞争获取同一个对象竞争同一把锁。

看下面这段代码:

//线程不安全,t1和t2尝试获取的是两个不同的对象,一个是类对象,一个是实例对象,获取的不是同一把锁,不存在锁冲突
public class Test {public static void main(String[] args) throws InterruptedException {A a = new A();//线程1获取到是static修饰的方法Thread t1 = new Thread(()->{A.fun1();try {Thread.sleep(10000);} catch (InterruptedException e) {e.printStackTrace();}});//线程2获取到的是非static修饰的方法Thread t2 = new Thread(()->{a.fun2();});t1.start();t2.start();t1.join();t2.join();}}
class A{static int count;//对静态方法加锁,就是对类对象加锁static synchronized void fun1(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是类方法"+count);}//对实例方法加锁,就是对实例对象加锁synchronized void  fun2(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是实例方法"+count);}
}

线程不安全,结果小于20000:
在这里插入图片描述

下面两段代码,多个线程就是在尝试获取同一锁
看下面这段代码:
t1、t2尝试获取同一把锁,实例对象锁

//t1和t2尝试获取到同一把锁,
public class Test {public static void main(String[] args) throws InterruptedException {A a = new A();//线程1获取到是非static修饰的方法Thread t1 = new Thread(()->{
//           A.fun1();a.fun2();});//线程2获取到的是非static修饰的方法Thread t2 = new Thread(()->{a.fun2();
//            A.fun1();});t1.start();t2.start();t1.join();t2.join();}}
class A{static int count;//对静态方法加锁,就是对类对象加锁static synchronized void fun1(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是类方法"+count);}//对实例方法加锁,就是对实例对象加锁synchronized void  fun2(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是实例方法"+count);}
}

看下面一段代码:
t1和t2尝试获取到同一把类锁

public class Test {public static void main(String[] args) throws InterruptedException {A a = new A();//线程1获取到是static修饰的方法Thread t1 = new Thread(()->{A.fun1();
//            a.fun2();});//线程2获取到的是static修饰的方法Thread t2 = new Thread(()->{
//           a.fun2();A.fun1();});t1.start();t2.start();t1.join();t2.join();}}
class A{static int count;//对静态方法加锁,就是对类对象加锁static synchronized void fun1(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是类方法"+count);}//对实例方法加锁,就是对实例对象加锁synchronized void  fun2(){for (int i = 0; i < 10000; i++) {count++;}System.out.println("我是实例方法"+count);}
}

四、总结

多线安全是否安全和不单单只是看synchronized修饰的属性,因为java里的任何对象都可以被synchronized修饰,关键在于多个线程是否是尝试获取相同的锁对象,如果是同一把锁就会发送锁冲突,线程安全。否则就不存在锁冲突,线程不安全。
所以不要被类锁和对象锁的名称给迷晕了,就看多个线程是否是在获取同一把锁,如果是同一个实例对象锁,线程安全;如果是同一个类对象锁,线程安全;如果是一个线程获取类锁,一个线程获取实例对象锁,不安全。

这篇关于类锁和实例对象锁你分清了吗?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/353859

相关文章

C# WinForms存储过程操作数据库的实例讲解

《C#WinForms存储过程操作数据库的实例讲解》:本文主要介绍C#WinForms存储过程操作数据库的实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、存储过程基础二、C# 调用流程1. 数据库连接配置2. 执行存储过程(增删改)3. 查询数据三、事务处

springboot security验证码的登录实例

《springbootsecurity验证码的登录实例》:本文主要介绍springbootsecurity验证码的登录实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录前言代码示例引入依赖定义验证码生成器定义获取验证码及认证接口测试获取验证码登录总结前言在spring

在java中如何将inputStream对象转换为File对象(不生成本地文件)

《在java中如何将inputStream对象转换为File对象(不生成本地文件)》:本文主要介绍在java中如何将inputStream对象转换为File对象(不生成本地文件),具有很好的参考价... 目录需求说明问题解决总结需求说明在后端中通过POI生成Excel文件流,将输出流(outputStre

tomcat多实例部署的项目实践

《tomcat多实例部署的项目实践》Tomcat多实例是指在一台设备上运行多个Tomcat服务,这些Tomcat相互独立,本文主要介绍了tomcat多实例部署的项目实践,具有一定的参考价值,感兴趣的可... 目录1.创建项目目录,测试文China编程件2js.创建实例的安装目录3.准备实例的配置文件4.编辑实例的

python+opencv处理颜色之将目标颜色转换实例代码

《python+opencv处理颜色之将目标颜色转换实例代码》OpenCV是一个的跨平台计算机视觉库,可以运行在Linux、Windows和MacOS操作系统上,:本文主要介绍python+ope... 目录下面是代码+ 效果 + 解释转HSV: 关于颜色总是要转HSV的掩膜再标注总结 目标:将红色的部分滤

Spring 中使用反射创建 Bean 实例的几种方式

《Spring中使用反射创建Bean实例的几种方式》文章介绍了在Spring框架中如何使用反射来创建Bean实例,包括使用Class.newInstance()、Constructor.newI... 目录1. 使用 Class.newInstance() (仅限无参构造函数):2. 使用 Construc

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

MyBatis-Plus中Service接口的lambdaUpdate用法及实例分析

《MyBatis-Plus中Service接口的lambdaUpdate用法及实例分析》本文将详细讲解MyBatis-Plus中的lambdaUpdate用法,并提供丰富的案例来帮助读者更好地理解和应... 目录深入探索MyBATis-Plus中Service接口的lambdaUpdate用法及示例案例背景

MyBatis-Plus中静态工具Db的多种用法及实例分析

《MyBatis-Plus中静态工具Db的多种用法及实例分析》本文将详细讲解MyBatis-Plus中静态工具Db的各种用法,并结合具体案例进行演示和说明,具有很好的参考价值,希望对大家有所帮助,如有... 目录MyBATis-Plus中静态工具Db的多种用法及实例案例背景使用静态工具Db进行数据库操作插入

Java实现将byte[]转换为File对象

《Java实现将byte[]转换为File对象》这篇文章将通过一个简单的例子为大家演示Java如何实现byte[]转换为File对象,并将其上传到外部服务器,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言1. 问题背景2. 环境准备3. 实现步骤3.1 从 URL 获取图片字节数据3.2 将字节数组