主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
Java

Java 基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

Java 控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

Java 面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

Java 内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

Java 文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

Java 错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

Java 多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

Java 同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

Java 网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

Java 集合

Java 集合框架
Java 集合接口

Java 接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

Java 数据结构

Java 数据结构
Java 枚举接口

Java 集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

Java 高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

Java 杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

集合

Java 集合框架
Java 集合接口

接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

数据结构

Java 数据结构
Java 枚举接口

集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

Java 可重入锁


上一章 下一章

ReentrantLock 是一个实现了 Lock 接口的类。它提供了同步功能的强大灵活性,这也是为什么它是 Java 中最常用的锁类。为了确保线程可靠且公平地工作是必要的。在这里,线程是大操作中的小子过程。在本文中,我们将学习 ReentrantLock 及其如何管理线程以便它们能高效地工作。

ReentrantLock 的工作原理

当多个线程试图访问共享资源时,ReentrantLock 通过 lock() 和 unlock() 方法限制一次只有一个线程访问。假设三个人同时尝试预订火车票,所有三个人都会尝试同时访问订票系统,可能会导致两个人最终预订了同一个座位。ReentrantLock 可以处理这种情况。

首先,所有人都会请求通过 tryLock() 方法获取订票系统。当一个人获得了订票系统后,则通过 lock() 方法限制特定座位的预订。在完成预订后,该人将调用 unlock() 方法释放已获取的锁。直到资源忙于处理时,其他人会在队列中等待他们的轮次,而在锁释放后,他们将进入可运行状态。

ReentrantLock 尝试公平地提供锁。我们可以设置线程可以获取锁的时间长度,并且它确保等待时间最长的线程可能首先获得锁。默认情况下,锁是非公平的,要使其公平,我们需要在其构造函数中传递布尔值 true。

语法

ReentrantLock nameOflock = new ReentrantLock(); 
// 默认为非公平锁

或者,

ReentrantLock nameOflock = new ReentrantLock(true); 
// 设置为公平锁

锁是显式的,并且可以以任意顺序锁定或解锁。单个线程可以多次请求锁,这就是锁的名字中有“Reentrant”(可重入)的原因。我们可以使用 getHoldCount() 方法来计算一个锁被获取了多少次。

没有 ReentrantLock 的多线程

以下示例说明了在上述代码中没有使用 ReentrantLock 的多线程。我们创建了一个类 Thrd,其中包含一个 operation() 方法来执行给定的任务。现在我们创建了三个线程类并调用 operation() 方法。在 main() 方法中定义了三个线程类的对象,并调用它们的 start() 方法以启动线程的执行。

没有 ReentrantLock 的多线程示例

package com.tutorialspoint;

class Thrd {
   static void operation(int data) {    
      for(int i = 1; i <= 4; i++) {
        System.out.println(data++);
      }
   }
}
class Thrd1 extends Thread {
   // 线程编号 1 
   public void run() {
     Thrd.operation(1);
     // 方法调用  
   }
}
class Thrd2 extends Thread {
   // 线程编号 2 
   public void run() {
     Thrd.operation(5);  
     // 方法调用
   }
}
class Thrd3 extends Thread {
   // 线程编号 3
   public void run() {
     Thrd.operation(10);  
     // 方法调用
   }
}
public class TestThread {
   public static void main(String args[]) {
     // 创建线程类的对象
     Thrd1 oprt1 = new Thrd1();
     Thrd2 oprt2 = new Thrd2();  
     Thrd3 oprt3 = new Thrd3();
     // 启动线程操作
     oprt1.start();
     oprt2.start();  
     oprt3.start();
   }
}

每次运行此程序都会产生不同的结果:

1
2
3
4
10
11
12
13
5
6
7
8

带有 ReentrantLock 的多线程

以下示例说明了上述代码中 ReentrantLock 的使用。我们在 Thrd 类内部定义了一个 ReentrantLock 对象。operation() 方法将 tryLock() 方法的布尔值存储到名为 lockAcquired 的变量中,该变量将检查是否有线程获取了锁。如果锁被获取,则使用 lock() 方法给予该线程锁,然后线程开始执行给定的任务。任务将在 try 块中执行,并在 finally 块中使用 unlock() 方法释放锁。现在我们创建了三个线程类并调用 operation() 方法。在 main() 方法中定义了三个线程类的对象,并调用它们的 start() 方法以启动线程的执行。

带有 ReentrantLock 的多线程示例

package com.tutorialspoint;

import java.util.concurrent.locks.ReentrantLock;

class Thrd {
   // 创建 ReentrantLock 类的对象
   private static ReentrantLock lockr = new ReentrantLock();
   static void operation(int data) {
     // 给予锁访问权
     boolean lockAcquired = lockr.tryLock(); 
     if (lockAcquired) {
       try {
         lockr.lock(); 
         // 给予线程锁
         for(int i = 1; i <= 4; i++) {
            System.out.println(data++);
         }
         // 检查锁计数
         System.out.println("Count of Lock: " + lockr.getHoldCount());
       } finally {
         lockr.unlock(); 
         // 解锁
       }
     } else {
       System.out.println("I am in else block");
     }
   }
}
class Thrd1 extends Thread {
   // 线程编号 1 
   public void run() {
     Thrd.operation(1);
     // 方法调用  
   }
}
class Thrd2 extends Thread {
   // 线程编号 2 
   public void run() {
     Thrd.operation(5);  
     // 方法调用
   }
}
class Thrd3 extends Thread {
   // 线程编号 3
   public void run() {
     Thrd.operation(10);  
     // 方法调用
   }
}
public class TestThread {
   public static void main(String args[]) {
     // 创建线程类的对象
     Thrd1 oprt1 = new Thrd1();
     Thrd2 oprt2 = new Thrd2();  
     Thrd3 oprt3 = new Thrd3();
     // 启动线程操作
     oprt1.start();
     oprt2.start();  
     oprt3.start();
   }
}

每次运行此程序都会产生不同的结果:

I am in else block
5
6
7
8
Count of Lock: 2
I am in else block

带有 ReentrantLock 为真值的多线程

以下示例说明了上述代码中 ReentrantLock 的使用。我们在 Thrd 类内部定义了一个带有公平值为 true 的 ReentrantLock 对象。operation() 方法将 tryLock() 方法的布尔值存储到名为 lockAcquired 的变量中,该变量将检查是否有线程获取了锁。如果锁被获取,则使用 lock() 方法给予该线程锁,然后线程开始执行给定的任务。任务将在 try 块中执行,并在 finally 块中使用 unlock() 方法释放锁。现在我们创建了三个线程类并调用 operation() 方法。在 main() 方法中定义了三个线程类的对象,并调用它们的 start() 方法以启动线程的执行。

带有 ReentrantLock 为真值的多线程示例

package com.tutorialspoint;

import java.util.concurrent.locks.ReentrantLock;

class Thrd {
   // 创建 ReentrantLock 类的对象
   private static ReentrantLock lockr = new ReentrantLock(true);
   static void operation(int data) {
     // 给予锁访问权
     boolean lockAcquired = lockr.tryLock(); 
     if (lockAcquired) {
       try {
         lockr.lock(); 
         // 给予线程锁
         for(int i = 1; i <= 4; i++) {
            System.out.println(data++);
         }
         // 检查锁计数
         System.out.println("Count of Lock: " + lockr.getHoldCount());
       } finally {
         lockr.unlock(); 
         // 解锁
       }
     } else {
       System.out.println("I am in else block");
     }
   }
}
class Thrd1 extends Thread {
   // 线程编号 1 
   public void run() {
     Thrd.operation(1);
     // 方法调用  
   }
}
class Thrd2 extends Thread {
   // 线程编号 2 
   public void run() {
     Thrd.operation(5);  
     // 方法调用
   }
}
class Thrd3 extends Thread {
   // 线程编号 3
   public void run() {
     Thrd.operation(10);  
     // 方法调用
   }
}
public class TestThread {
   public static void main(String args[]) {
     // 创建线程类的对象
     Thrd1 oprt1 = new Thrd1();
     Thrd2 oprt2 = new Thrd2();  
     Thrd3 oprt3 = new Thrd3();
     // 启动线程操作
     oprt1.start();
     oprt2.start();  
     oprt3.start();
   }
}

每次运行此程序都会产生不同的结果:

I am in else block
I am in else block
5
6
7
8
Count of Lock: 2
上一章 下一章
阅读号二维码

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图