Java 教程
- Java - 首页
- Java - 概述
- Java - 历史
- Java - 特性
- Java 与 C++
- 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 - OOPs 概念
- 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 - try-catch 块
- Java - try-with-resources
- Java - 多重捕获块
- Java - 嵌套 try 块
- Java - Finally 块
- Java - throw 异常
- Java - 异常传播
- Java - 内置异常
- Java - 自定义异常
Java 多线程
- Java - 多线程
- Java - 线程生命周期
- Java - 创建线程
- Java - 启动线程
- Java - 合并线程
- 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 - Lambda 表达式
- Java - 发送邮件
- Java - Applet 基础
- Java - Javadoc 注释
- Java - 自动装箱和拆箱
- Java - 文件不匹配方法
- Java - REPL (JShell)
- Java - 多版本 Jar 文件
- Java - 私有接口方法
- Java - 内部类菱形运算符
- Java - 多分辨率图像 API
- Java - 集合工厂方法
- Java - 模块系统
- Java - Nashorn JavaScript
- Java - Optional 类
- Java - 方法引用
- Java - 函数式接口
- Java - 默认方法
- Java - Base64 编码解码
- Java - Switch 表达式
- Java - Teeing 收集器
- Java - 微基准测试
- Java - 文本块
- Java - 动态 CDS 归档
- Java - Z 垃圾收集器 (ZGC)
- Java - 空指针异常
- Java - 打包工具
- Java - 密封类
- Java - 记录类
- Java - 隐藏类
- Java - 模式匹配
- Java - 紧凑数字格式化
- Java - 垃圾回收
- Java - JIT 编译器
Java 杂项
- Java - 递归
- Java - 正则表达式
- Java - 序列化
- Java - 字符串
- Java - 进程 API 改进
- Java - 流 API 改进
- Java - 增强的 @Deprecated 注解
- Java - CompletableFuture API 改进
- Java - 流
- Java - 日期时间 Api
- Java 8 - 新特性
- Java 9 - 新特性
- Java 10 - 新特性
- Java 11 - 新特性
- Java 12 - 新特性
- Java 13 - 新特性
- Java 14 - 新特性
- Java 15 - 新特性
- Java 16 - 新特性
Java API 和框架
Java 类引用
- Java - Scanner
- Java - 数组
- Java - 字符串
- Java - Date
- Java - ArrayList
- Java - Vector
- Java - Stack
- Java - PriorityQueue
- Java - LinkedList
- Java - ArrayDeque
- Java - HashMap
- Java - LinkedHashMap
- Java - WeakHashMap
- Java - EnumMap
- Java - TreeMap
- Java - IdentityHashMap
- Java - HashSet
- Java - EnumSet
- Java - LinkedHashSet
- Java - TreeSet
- Java - BitSet
- Java - Dictionary
- Java - Hashtable
- Java - Properties
- Java - Collection
- Java - Array
Java 有用资源
Java - 块同步
当我们在 Java 程序中启动 两个或多个线程 时,可能会出现多个线程尝试访问相同资源的情况,最终由于并发问题而产生不可预见的结果。例如,如果多个线程尝试写入同一个文件,则它们可能会损坏数据,因为其中一个线程可能会覆盖数据,或者当一个线程正在打开同一个文件时,另一个线程可能正在关闭同一个文件。
因此,需要同步多个线程的操作,并确保在给定时间点只有一个线程可以访问资源。这是使用称为监视器的概念实现的。Java 中的每个对象都与一个监视器关联,线程可以锁定或解锁该监视器。一次只有一个线程可以持有监视器的锁。
Java 中的块同步
Java 编程语言提供了一种非常方便的方法来创建线程并使用同步块 同步 它们的作业。您将共享资源保留在此块中。
语法
以下是 synchronized 语句的一般形式:
synchronized(objectidentifier) { // Access shared variables and other shared resources }
这里,objectidentifier 是对一个对象的引用,其锁与 synchronized 语句表示的监视器关联。现在我们将看两个示例,我们将在其中使用两个不同的线程打印计数器。当线程不同步时,它们打印的计数器值不是按顺序的,但是当我们将计数器放在 synchronized() 块内打印时,它会为两个线程按顺序打印计数器。
无同步的多线程示例
这是一个简单的示例,它可能打印或不打印按顺序的计数器值,并且每次我们运行它时,它都会根据 CPU 对线程的可用性产生不同的结果。
示例
package com.tutorialspoint; class PrintDemo { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread interrupted."); } } } class ThreadDemo extends Thread { private Thread t; private String threadName; PrintDemo printDemo; ThreadDemo( String name, PrintDemo pd) { threadName = name; printDemo = pd; } public void run() { printDemo.printCount(); System.out.println("Thread " + threadName + " exiting."); } public void start () { System.out.println("Starting " + threadName ); if (t == null) { t = new Thread (this, threadName); t.start (); } } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); ThreadDemo t1 = new ThreadDemo( "Thread - 1 ", printDemo ); ThreadDemo t2 = new ThreadDemo( "Thread - 2 ", printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
输出
每次运行此程序都会产生不同的结果:
Starting Thread - 1 Starting Thread - 2 Counter --- 5 Counter --- 5 Counter --- 4 Counter --- 4 Counter --- 3 Counter --- 3 Counter --- 2 Counter --- 2 Counter --- 1 Counter --- 1 Thread Thread - 1 exiting. Thread Thread - 2 exiting.
块级同步的多线程示例
这是同一个示例,它按顺序打印计数器值,并且每次我们运行它时,它都会产生相同的结果。我们在块上放置了 synchronized 关键字,以便计数器增量代码现在在方法执行期间根据对象被锁定。我们使用当前对象作为锁,将其作为参数传递给 synchronized 块。
示例
package com.tutorialspoint; class PrintDemo { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread interrupted."); } } } class ThreadDemo extends Thread { private Thread t; private String threadName; PrintDemo printDemo; ThreadDemo( String name, PrintDemo pd) { threadName = name; printDemo = pd; } public void run() { synchronized(printDemo) { printDemo.printCount(); } System.out.println("Thread " + threadName + " exiting."); } public void start () { System.out.println("Starting " + threadName ); if (t == null) { t = new Thread (this, threadName); t.start (); } } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); ThreadDemo t1 = new ThreadDemo( "Thread - 1 ", printDemo ); ThreadDemo t2 = new ThreadDemo( "Thread - 2 ", printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
输出
每次运行此程序都会产生相同的结果:
Starting Thread - 1 Starting Thread - 2 Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread - 1 exiting. Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread - 2 exiting.
方法级同步的多线程示例
这是同一个示例,它按顺序打印计数器值,并且每次我们运行它时,它都会产生相同的结果。这次我们在方法上放置了 synchronized 关键字,以便整个方法在方法执行期间根据对象被锁定。
示例
package com.tutorialspoint; class PrintDemo extends Thread { public void printCount() { try { for(int i = 5; i > 0; i--) { Thread.sleep(50); System.out.println("Counter --- " + i ); } } catch (Exception e) { System.out.println("Thread " + Thread.currentThread().getName()+" interrupted."); } } public synchronized void run() { printCount(); System.out.println("Thread " + Thread.currentThread().getName() + " exiting."); } } public class TestThread { public static void main(String args[]) { PrintDemo printDemo = new PrintDemo(); Thread t1 = new Thread(printDemo ); Thread t2 = new Thread(printDemo ); t1.start(); t2.start(); // wait for threads to end try { t1.join(); t2.join(); } catch ( Exception e) { System.out.println("Interrupted"); } } }
输出
每次运行此程序都会产生相同的结果:
Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread-1 exiting. Counter --- 5 Counter --- 4 Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread-2 exiting.