java 并发编程之同步器

2024-05-09 08:08
文章标签 java 并发 编程 同步器

本文主要是介绍java 并发编程之同步器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

 在讨论Java的锁的时候,有一个概念很重要,那就是同步器,同步器可以想象成一种互斥量,当一个线程在进入临界区之前先要获得互斥量,同样,在线程获得同步状态之前也是需要获取到同步器。今天看到《java并发编程的艺术》第五章的时候觉得蛮有收获,所以这一篇算是整理下看书思路。

 

首先,我们来看一下Lock的接口

 

public interface Lock {
void lock();
boolean tryLock();
void unlock();
Condition newCondition();
}

 这个接口只是提供了几个常见的方法,而没有实现,当我们需要使用锁的时候会实现这个接口的lock和unlock方法,在这些方法的实现就是需要用到队列同步器 .AbstractQueuedSynchronizer, AbstractQueuedSynchronizer这个类提供了很多的同步方法,主要可分为三类:

1,状态追踪方法

       getState()   :获取当前状态

       setState(int ):设置当前状态

       compareAndSetState(int expect,int update);  CAS如果当前是expect值则设置其值为update,CAS具有同步功能,因为它具有volatile的读写特性,保证编译器不会对这一段代码进行重排序

2,独占锁同步

       acquire(int arg) 获取锁

       acquireInterruptibly(int arg)  中断

       tryAcquireNanos(int arg,long nanos)  尝试获取锁,并且如果超时则放弃

       

       boolean release(int arg)  释放锁

       

3,共享锁同步 共享锁和独占所方法解释差不多,只是共享锁能够同时让多个线程执行,独占锁只能单独的线程获取同步状态后执行,其余线程会进入等待队列

       acquireShared(int arg)

       acquireSharedInterruptibly(int arg)

      tryAcqiuireSharedNanos(int arg,long nanos)

 

      boolean releaseShared(int arg);      

     

AbstractQueuedSynchronizer这个同步队列在很多的并行类或者容器中都能看到身影,因为它是负责很多的同步处理的关键,一个线程在进入临界区之前需要执行acquire方法获得同步机会,而acquire方法先要tryAcquire()如果能够获取则进入执行,如果不能获取则进入等待队列(acquire有比较复杂的方法使得进入等待队列能够实现同步,而且保持FIFO的算法)。

 

我们可以自定义自己的锁和同步器,通过继承AbstractQueuedSynchronizer这个类实现自己的同步器,而因为AbstractQueuedSynchronizer的acquire和release方法的都要执行tryAcquire和tryRelease方法,因此我们只需要重写这两个方法就能实现自定义的同步算法。

 

首先,我们来实现一个自定义的独占锁

 

package com.luchi.concurrencyImpl;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
public class Mutex implements Lock{
//自定义内部类同步器
private static class Sync extends AbstractQueuedSynchronizer{
@Override
protected boolean isHeldExclusively(){
return getState()==1;
}
@Override
public boolean tryAcquire(int acquires){
if(compareAndSetState(0, 1)){
setExclusiveOwnerThread(Thread.currentThread());
return true;
}
return false;
}
@Override
public boolean tryRelease(int release){
if(getState()==0){
throw new IllegalMonitorStateException();
}
setExclusiveOwnerThread(null);
setState(0);
return true;
}
Condition newCondition(){return new ConditionObject();}
}
private final Sync sync=new Sync();
@Override
public void lock() {
// TODO Auto-generated method stub
sync.acquire(1);
}
@Override
public void unlock() {
// TODO Auto-generated method stub
sync.release(1);
}
@Override
public void lockInterruptibly() throws InterruptedException {
// TODO Auto-generated method stub
sync.acquireInterruptibly(1);
}
@Override
public boolean tryLock() {
// TODO Auto-generated method stub
return sync.tryAcquire(1);
}
@Override
public boolean tryLock(long time, TimeUnit unit)
throws InterruptedException {
// TODO Auto-generated method stub
return sync.tryAcquireNanos(1,unit.toNanos(time));
}
@Override
public Condition newCondition() {
// TODO Auto-generated method stub
return sync.newCondition();
}
}

 

接着我们来定义一段共享锁

 

package com.luchi.concurrencyImpl;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
public class TwinsLock implements Lock{
private final Sync sync=new Sync(2);
//自定义队列同步器
private static final class Sync extends AbstractQueuedSynchronizer{
//初始化队列同步器的
Sync(int count){
if(count<=0){
throw new  IllegalArgumentException("count must larger than zero");
}
setState(count);
}
@Override
public int tryAcquireShared(int reduceCount){
while(true){
int current=getState();
int newCount=current-reduceCount;
if(newCount<0 || compareAndSetState(current,newCount)){
return newCount;
}
}
}
@Override
public boolean tryReleaseShared(int returnCount){
while(true){
int current=getState();
int newCount=current+returnCount;
if(compareAndSetState(current, newCount)){
return true;
}
}
}
}
@Override
public void lock() {
// TODO Auto-generated method stub
sync.acquireShared(1);
}
@Override
public void lockInterruptibly() throws InterruptedException {
// TODO Auto-generated method stub
}
@Override
public boolean tryLock() {
// TODO Auto-generated method stub
return false;
}
@Override
public boolean tryLock(long time, TimeUnit unit)
throws InterruptedException {
// TODO Auto-generated method stub
return false;
}
@Override
public void unlock() {
// TODO Auto-generated method stub
sync.releaseShared(1);
}
@Override
public Condition newCondition() {
// TODO Auto-generated method stub
return null;
}
}

  可以看到,无论是共享锁合适独占锁,在自定义的时候只需要重写tryAcquire(或者tryAcquireShare)方法和tryRelease(或者tryReleaseShared)方法即可,独占锁因为“互斥量”为1所以不需要初始化,共享锁则可以自定义互斥量的多少,也就是定义多少个线程能够同时进入同步区。

 

另外我们也可以看到,锁和同步器的分离降低了两者之间的耦合度,也方便程序开发人员进行模块化定义和分工,这符合软件工程的思维

 

 

这篇关于java 并发编程之同步器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

springboot集成easypoi导出word换行处理过程

《springboot集成easypoi导出word换行处理过程》SpringBoot集成Easypoi导出Word时,换行符n失效显示为空格,解决方法包括生成段落或替换模板中n为回车,同时需确... 目录项目场景问题描述解决方案第一种:生成段落的方式第二种:替换模板的情况,换行符替换成回车总结项目场景s

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

基于 Cursor 开发 Spring Boot 项目详细攻略

《基于Cursor开发SpringBoot项目详细攻略》Cursor是集成GPT4、Claude3.5等LLM的VSCode类AI编程工具,支持SpringBoot项目开发全流程,涵盖环境配... 目录cursor是什么?基于 Cursor 开发 Spring Boot 项目完整指南1. 环境准备2. 创建

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq