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

news/2024/11/29 20:44:56/

系列文章目录

文章目录

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


前言

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.ppmy.cn/news/1198344.html

相关文章

在本地安装LLAMA 2

方法一&#xff1a; Meta已将llama2开源&#xff0c;任何人都可以通过在meta ai上申请并接受许可证、提供电子邮件地址来获取模型。 Meta 将在电子邮件中发送下载链接。 下载llama2 获取download.sh文件&#xff0c;将其存储在mac上打开mac终端&#xff0c;执行 chmod x ./do…

lv9 嵌入式开发 数据库sqlite

1 数据库基本概念 数据&#xff08;Data&#xff09; 能够输入计算机并能被计算机程序识别和处理的信息集合 数据库 &#xff08;Database&#xff09; 数据库是在数据库管理系统管理和控制之下&#xff0c;存放在存储介质上的数据集合 2 常用的数据库 大型数据库…

802.11 CSMA/CA协议

《计算机网络自顶向下》P351的总结提炼

vue的rules验证失效,部分可以部分又失效的原因

vue的rules验证失效,部分可以部分又失效的原因 很多百度都有,但是我这里遇到了一个特别的,那就是prop没有写全,导致验证某一个失效 例子: 正常写法 el-form-item....多个省略<el-form-item label"胶币" prop"cost"><el-input v-model"form.…

【Spring Security】Spring Security 前后端分离认证

我们初步引入了Spring Security&#xff0c;并使用其默认生效的HTTP基本认证来保护URL资源&#xff0c;本章我们使用表单认证来保护URL资源。 前后端分离模式 表单登录配置模块提供了successHandler&#xff08;&#xff09;和failureHandler&#xff08;&#xff09;两个方法…

vite配置.env环境变量文件,开发环境,测试环境,预发布环境,生产环境

在vue2&#xff0c;用的vue-cli脚手架搭建项目&#xff0c;cli用的是webpack 当你yarn dev时&#xff0c;命令会启动package.json中的dev键名的值&#xff0c;也就是后面的一行命令 这时浏览器会去识别你是开发环境还是生产环境&#xff0c;其实windows是不能直接识别你是开发…

33 mysql find_in_set 的实现

前言 这里我们主要是来探讨一下 mysql 中 in 的使用, find_in_set 的使用 这两者 在我们实际应用中应该也是 非常常用的了 测试数据表如下 CREATE TABLE tz_test (id int(11) unsigned NOT NULL AUTO_INCREMENT,field1 varchar(16) DEFAULT NULL,field2 varchar(16) DEFAU…

Python 用户输入和字符串格式化指南

Python 允许用户输入数据。这意味着我们可以向用户询问输入。在 Python 3.6 中&#xff0c;使用 input() 方法来获取用户输入。在 Python 2.7 中&#xff0c;使用 raw_input() 方法来获取用户输入。以下示例要求用户输入用户名&#xff0c;并在输入用户名后将其打印在屏幕上&am…