Java - 线程同步
为什么需要线程同步?
当我们在程序中启动两个或多个线程时,可能会出现多个线程试图访问同一资源的情况,最终由于并发问题导致不可预见的结果。例如,如果多个线程试图写入同一个文件,它们可能会破坏数据,因为其中一个线程可能会覆盖数据,或者当一个线程正在打开同一文件时,另一个线程可能同时正在关闭该文件。
因此,需要同步多个线程的操作,确保在任何给定时间点只有一个线程可以访问该资源。这是通过一种称为monitors的概念来实现的。Java 中的每个对象都关联着一个 monitor,线程可以锁定或解锁该 monitor。任何时候只有一个线程可以持有 monitor 的锁。
Java 中的线程同步
Java 编程语言提供了一种非常便捷的方式来创建线程并同步它们的任务,即使用 synchronized 块。将共享资源保存在这个块中。以下是 synchronized 语句的一般形式:
语法
synchronized(objectidentifier) {
// 访问共享变量和其他共享资源
}
在这里,objectidentifier 是对对象的引用,该对象的锁与 synchronized 语句所代表的 monitor 相关联。现在我们来看两个示例,我们将使用两个不同的线程打印一个计数器。当线程未同步时,它们打印的计数器值不是顺序的,但当我们将计数器放在 synchronized() 块中打印时,两个线程都会非常有序地打印计数器。
没有线程同步的多线程示例
以下是一个简单的示例,它可能不会按顺序打印计数器值,每次运行时都会根据 CPU 对线程的可用性产生不同的结果。
示例
class PrintDemo {
public void printCount() {
try {
for(int i = 5; i > 0; i--) {
System.out.println("Counter --- " + i );
}
} catch (Exception e) {
System.out.println("Thread interrupted.");
}
}
}
class ThreadDemo extends Thread {
private Thread t;
private String threadName;
PrintDemo PD;
ThreadDemo( String name, PrintDemo pd) {
threadName = name;
PD = pd;
}
public void run() {
PD.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 PD = new PrintDemo();
ThreadDemo T1 = new ThreadDemo( "Thread - 1 ", PD );
ThreadDemo T2 = new ThreadDemo( "Thread - 2 ", PD );
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 --- 5 Counter --- 2 Counter --- 1 Counter --- 4 Thread Thread - 1 exiting. Counter --- 3 Counter --- 2 Counter --- 1 Thread Thread - 2 exiting.
使用线程同步的多线程示例
这是一个相同的示例,它会按顺序打印计数器值,并且每次运行都会产生相同的结果。
示例
class PrintDemo {
public void printCount() {
try {
for(int i = 5; i > 0; i--) {
System.out.println("Counter --- " + i );
}
} catch (Exception e) {
System.out.println("Thread interrupted."); // 线程被中断。
}
}
}
class ThreadDemo extends Thread {
private Thread t;
private String threadName;
PrintDemo PD;
ThreadDemo( String name, PrintDemo pd) {
threadName = name;
PD = pd;
}
public void run() {
synchronized(PD) {
PD.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 PD = new PrintDemo();
ThreadDemo T1 = new ThreadDemo( "Thread - 1 ", PD );
ThreadDemo T2 = new ThreadDemo( "Thread - 2 ", PD );
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.
java_multithreading.htm