
在多线程编程中,当多个线程同时访问或修改共享资源时,可能会出现竞态条件(race condition),导致数据不一致或操作中断。例如,一个线程正在执行一个多步操作(如打印一条完整的日志信息),而另一个线程突然插入并执行自己的操作,就会导致输出混乱或不完整。开发者有时会尝试通过设置线程优先级来解决这类问题,期望高优先级的线程能够优先完成其任务而不被中断。然而,线程优先级在java中通常不可靠,其行为高度依赖于底层操作系统和jvm实现,并不能保证严格的执行顺序或互斥访问。
Java提供了Thread.setPriority()方法来设置线程的优先级,范围从Thread.MIN_PRIORITY到Thread.MAX_PRIORITY。理论上,高优先级的线程会比低优先级的线程获得更多的CPU时间片。然而,在实际应用中,线程优先级存在以下局限:
因此,对于需要确保操作原子性或互斥访问共享资源的场景,线程优先级并非合适的解决方案。
为了确保在多线程环境下对共享资源的访问是互斥的,Java提供了synchronized关键字。synchronized可以用于方法或代码块,它通过使用一个内部锁机制来保证在同一时间只有一个线程可以执行被synchronized保护的代码。
当一个线程进入一个synchronized代码块或方法时,它会尝试获取与该代码块或方法关联的锁。如果锁已被其他线程持有,当前线程就会被阻塞,直到锁被释放。一旦当前线程获取到锁,它就可以独占地执行synchronized代码,直到执行完毕并释放锁。
立即学习“Java免费学习笔记(深入)”;
对于需要保护特定代码段免受并发访问的场景,通常使用synchronized代码块。它需要一个对象作为锁。
示例代码:
public class PrinterManager {
// 定义一个共享的锁对象,通常声明为final,避免被重新赋值
public static final Object MY_LOCK = new Object();
public void printMessage1(String message) {
// 使用MY_LOCK对象作为锁,确保printMessage1方法中的打印操作是互斥的
synchronized (MY_LOCK) {
System.out.print("[线程 " + Thread.currentThread().getName() + "] 开始打印: ");
try {
// 模拟打印过程中的耗时操作
Thread.sleep(50);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
System.out.println(message + " (结束)");
}
}
public void printMessage2(String message) {
// 同样使用MY_LOCK对象作为锁,确保与printMessage1互斥
synchronized (MY_LOCK) {
System.out.print("[线程 " + Thread.currentThread().getName() + "] 开始打印: ");
try {
Thread.sleep(50);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
System.out.println(message + " (结束)");
}
}
public static void main(String[] args) {
PrinterManager manager = new PrinterManager();
// 创建并启动多个线程,它们将尝试同时调用printMessage方法
for (int i = 0; i < 5; i++) {
final int taskId = i;
new Thread(() -> {
manager.printMessage1("任务A-" + taskId);
}, "Thread-" + i + "-A").start();
new Thread(() -> {
manager.printMessage2("任务B-" + taskId);
}, "Thread-" + i + "-B").start();
}
}
}在上述示例中:
当需要同步整个方法时,可以直接在方法声明上使用synchronized关键字。对于实例方法,锁是方法所属的实例对象(this);对于静态方法,锁是方法所属的类的Class对象。
public class SynchronizedMethodExample {
// 实例方法,锁是SynchronizedMethodExample的实例对象
public synchronized void instanceMethod() {
System.out.println("实例方法 - 线程: " + Thread.currentThread().getName());
// ... 临界区代码 ...
}
// 静态方法,锁是SynchronizedMethodExample.class对象
public static synchronized void staticMethod() {
System.out.println("静态方法 - 线程: " + Thread.currentThread().getName());
// ... 临界区代码 ...
}
}注意: 当使用synchronized方法时,需要确保所有需要互斥访问的方法都使用相同的锁(即同一个实例对象或同一个Class对象)。
选择合适的锁对象:
锁的粒度: 锁的粒度应尽可能小,只保护真正需要同步的代码块。过大的锁粒度会降低并发性,因为线程会花费更多时间等待锁的释放。
避免死锁: 当两个或多个线程互相持有对方所需的锁时,就会发生死锁。设计同步机制时应仔细考虑锁的获取顺序,尽量保持一致。
性能考量: synchronized是Java提供的开箱即用的同步机制,使用方便。但在高并发场景下,其性能可能不如java.util.concurrent.locks.ReentrantLock等显式锁。ReentrantLock提供了更细粒度的控制,如公平锁、条件变量等,但使用起来也更复杂。
在Java多线程编程中,解决因并发访问共享资源导致的操作中断和数据不一致问题,应优先采用synchronized关键字或java.util.concurrent包下的锁机制。线程优先级并非用于实现严格的互斥或控制执行顺序的可靠方法。通过正确使用synchronized代码块和共享锁对象,我们可以有效地保护临界区,确保关键操作的原子性,从而构建健壮、可靠的多线程应用程序。
以上就是Java多线程并发控制:使用synchronized关键字实现互斥访问的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号