-
死锁:发生在两个或更多线程互相等待对方持有的资源,导致所有的线程都无法进行下去。避免死锁的一个常见方法是遵循资源顺序访问,将系统中的资源排序,并约定每个线程都按序请求资源。
-
竞态条件:两个或更多线程同时访问和修改共享资源,结果取决于线程运行的精确时序,可能导致不可预期的结果。避免竞态条件的一个方法是使用互斥量(或锁)来确保同一时刻只有一个线程能够访问特定的资源。
-
线程不安全:线程不安全主要是指当多个线程访问和修改相同的数据时,可能会导致数据的不一致。通过线程同步和使用线程安全的数据结构,如
java.util.concurrent
包中提供的数据结构,可以解决线程不安全的问题。 -
内存可见性:在并发编程中,由于线程之间的缓存不一致,一个线程修改的共享变量可能无法立即对其他线程可见。Java中的
volatile
关键字可以确保所有线程都能看到共享变量的最新值。
1、锁和同步化代码块来避免线程不安全和竞态条件:
public class Counter{
private int counter;
// 用 synchronized 关键字来保护共享资源
public synchronized void increment(){
counter++;
}
}
2、使用java.util.concurrent
包下的线程安全集合来避免线程不安全:
// 使用线程安全的 ConcurrentHashMap
Map<String, String> safeMap = new ConcurrentHashMap<>();
3、使用volatile
关键字来保证内存可见性:
public class SharedData {
// 使用 volatile 关键字来保证内存可见性
volatile int sharedCounter = 0;
}
4、避免死锁的一种方式是总是按相同的顺序获取锁,下面是一个例子:
class OrderedLock {
private final Object firstLock = new Object();
private final Object secondLock = new Object();
public void doSomething() {
synchronized (firstLock) {
synchronized (secondLock) {
// do something
}
}
}
}