当前位置: 首页 > news >正文

java 线程安全

内部锁

在 Java 中,每个对象都有一个内部锁,也称为监视器锁或对象锁。内部锁是通过在代码块或方法前加上 synchronized 关键字来实现的。当一个线程执行一个带有 synchronized 关键字的方法或代码块时,它必须先获得该对象的内部锁,才能执行该方法或代码块。

同步代码块

synchronized(锁对象){

}

把同步代码块中的代码锁住,其中的代码只能让一个线程执行(在同一时间内)

必须保证 锁对象 总是同一个对象

想象 锁对象 代表一道门,没有线程进入时,门开着,一旦有线程A进入之后就会关上这扇门,其他线程就不能从这扇门进入同步代码块,当线程A执行完同步代码块中的内容离开就会重新打开这道门,其他线程才能够进入。

但如果其他线程B来的时候拿到的是另一个 锁对象 ,另一个 锁对象就好是另一扇门b, 这一扇门没有被关闭, 后来的线程B从 门b 就可以进入同步代码块,因此产生了不同步问题。

 同步方法

public synchronized void increment() { count++; } 

同步方法的锁为调用方法的对象 this

静态方法不能使用this,那就是 this.class

Lock

Java中的Lock是一种用于控制多个线程访问共享资源的机制。与传统的同步块相比,Lock提供了更细粒度的线程控制,并且可以提供更多的灵活性和性能优势。

  1. Lock的定义与基本用法

Lock是Java提供的一种线程同步机制。与synchronized不同,Lock需要程序员手动申请和释放锁。Java中提供了两种类型的Lock:ReentrantLock和ReentrantReadWriteLock.ReadLock/WriteLock。其中,ReentrantLock是一个可重入的互斥锁,而ReentrantReadWriteLock则支持读写锁。

Lock lock = new ReentrantLock();
lock.lock();
try {// 在锁定状态下执行线程安全的代码
} finally {lock.unlock();
}

上述代码中,首先创建了一个ReentrantLock对象,然后通过调用lock()方法获得了该锁的控制权,进而执行了线程安全的代码。最后,通过调用unlock()方法释放了该锁。

需要注意的是,为了避免死锁的发生,我们应该总是使用try-finally语句块来保证在任何情况下都能释放锁。

2 可中断锁

在某些情况下,线程需要能够响应中断请求,这时我们可以使用可中断锁。当一个线程正在等待获取锁的时候,如果接收到了中断请求,它可以选择放弃获取锁,而不是一直等待下去。

Lock lock = new ReentrantLock();
try {lock.lockInterruptibly();try {// 在锁定状态下执行线程安全的代码} finally {lock.unlock();}
} catch (InterruptedException e) {// 处理中断异常
}

在上述代码中,我们使用了lockInterruptibly()方法来获得可中断锁的控制权。如果在等待获取锁的过程中,线程接收到了中断请求(线程的interrupt()方法,向其发送中断请求),就会抛出InterruptedException异常。在catch语句块中,我们可以处理该异常,并做出相应的处理。

读写锁

Java 中的读写锁是一种特殊的锁,用于控制多个线程对共享资源的读和写。读写锁分为两种类型:读锁和写锁。读锁允许多个线程同时读取共享资源,而写锁只允许一个线程写入共享资源。如果一个线程获取了写锁,则其他线程不能读取或写入共享资源。

Java 中的 java.util.concurrent.locks.ReentrantReadWriteLock 类提供了读写锁的实现。该类有两个方法:readLock()writeLock(),分别返回读锁和写锁。获取读锁的线程可以同时读取共享资源,获取写锁的线程必须等待所有读锁释放后才能写入共享资源。

import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;public class SharedData {private final ReadWriteLock lock = new ReentrantReadWriteLock();private int data = 0;public void writeData(int newData) {lock.writeLock().lock();try {data = newData;} finally {lock.writeLock().unlock();}}public int readData() {lock.readLock().lock();try {return data;} finally {lock.readLock().unlock();}}
}

在上面的示例代码中,SharedData 类使用 ReadWriteLock 来控制对共享数据 data 的读写访问。writeData 方法获取写锁,修改 data 变量的值,然后释放写锁。readData 方法获取读锁,读取 data 变量的值,然后释放读锁。由于多个线程可以同时获取读锁,因此读操作可以并发进行,而写操作需要独占锁,因此写操作只能一个一个地进行。

读写锁适用于对于某个数据结构,读的操作远远多于写操作的场景。如果读写操作的次数差不多,使用读写锁的效果并不好,还不如使用内部锁。

http://www.lryc.cn/news/65338.html

相关文章:

  • 移动硬盘修复后文件丢失恢复方法
  • 直线飙升到10万+star的AutoGpt,有多强?帮我写了个网页!
  • rk3568平台调试typec口实现uvc输出,网络共享等功能
  • java基础知识——26.反射
  • 【容器化】Docker 简介和安装
  • 性能测试场景分析并设计?超细案例讲解,看这篇就够了
  • JAVA9新特性
  • ( 数组和矩阵) 645. 错误的集合 ——【Leetcode每日一题】
  • 2023年全国最新道路运输从业人员精选真题及答案63
  • Kettle安装与使用
  • C51 - DS18B20
  • 手把手教你使用vue2搭建微前端micro-app
  • DDR3(MIG核配置官方demoFPGA代码实现及仿真)
  • 传奇人物《周兴和》书连载之67 不辱神圣的使命
  • Spring框架中的单例Beans是线程安全的么?
  • AI脚本插件开发-链接图自动建立档名-插件制作源码-illustrator插件开发
  • rust智能指针
  • Git、Gitee、Github、Gitlab区别与联系
  • 接口优化的策略
  • android 隐藏底部虚拟按键
  • 基于电流控制的并网逆变器(Simulink)
  • learn_C_deep_9 (汇编角度理解return的含义、const 的各种应用场景)
  • 基于深度学习的OCR技术
  • 『python爬虫』09. bs4实战之下载精美壁纸(保姆级图文)
  • 【Linux学习】多线程——线程控制 | 线程TCB
  • Node 10 接口
  • 大型互联网企业大流量高并发电商领域核心项目已上线(完整流程+项目白皮书)
  • 汇编语言学习笔记六
  • 多商户商城系统-v2.2.3版本发布
  • 科研人必看入门攻略(收藏版)