目录
1. 信号灯法(Synchronized实现)
public class PCDemo {
public static void main(String[] args) {
Data data = new Data();
new Thread(()->{
for (int i = 0; i < 10; i++) {
try {
data.produce();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"A").start();
new Thread(()->{
for (int i = 0; i < 10; i++) {
try {
data.consum();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"B").start();
new Thread(()->{
for (int i = 0; i < 10; i++) {
try {
data.produce();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"C").start();
new Thread(()->{
for (int i = 0; i < 10; i++) {
try {
data.consum();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"D").start();
}
}
//判断等待、业务、通知
class Data {
private int number = 0;
//+1
public synchronized void produce() throws InterruptedException {
//1.判断等待
while (number != 0) { // !!!!!!注意解决虚假唤醒,必须用while,不能用if
wait();
}
//2.业务
number++;
System.out.println(Thread.currentThread().getName() + "=>" + number);
//3.通知
this.notifyAll();
}
//-1
public synchronized void consum() throws InterruptedException {
//1.判断等待
while (number == 0) { // !!!!!!注意解决虚假唤醒,必须用while,不能用if
wait();
}
//2.业务
number--;
System.out.println(Thread.currentThread().getName() + "=>" + number);
//3.通知
this.notifyAll();
}
}
解决虚假唤醒![](https://img-blog.csdnimg.cn/5c06d15045cf4c97bf43e9233393ede3.png)
2. 信号灯法(Lock实现)
2.1 基本实现
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class Data2 {
private int number = 0;
private Lock lock = new ReentrantLock();
private Condition condition = lock.newCondition();
//+1
public void produce() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number != 0) {// !!!!!!注意解决虚假唤醒,必须用while,不能用if
condition.await();
}
//2.业务
number++;
System.out.println(Thread.currentThread().getName() + "=>" + number);
//3.通知
condition.signalAll();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
//-1
public void consum() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number == 0) {
condition.await();
}
//2.业务
number--;
System.out.println(Thread.currentThread().getName() + "=>" + number);
//3.通知
condition.signalAll();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}
2.2 Condition精准通知
public class PCDemo3 {
public static void main(String[] args) {
Data3 data = new Data3();
new Thread(() -> {
for (int i = 0; i < 10; i++) {
try {
data.produceA();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "A").start();
new Thread(() -> {
for (int i = 0; i < 10; i++) {
try {
data.consumB();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "B").start();
new Thread(() -> {
for (int i = 0; i < 10; i++) {
try {
data.produceC();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "C").start();
new Thread(() -> {
for (int i = 0; i < 10; i++) {
try {
data.consumD();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, "D").start();
}
}
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class Data3 {
private int number = 1;
private Lock lock = new ReentrantLock();
private Condition condition1 = lock.newCondition();
private Condition condition2 = lock.newCondition();
private Condition condition3 = lock.newCondition();
private Condition condition4 = lock.newCondition();
//执行顺序:A->B->C->D
//+1
public void produceA() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number != 1) {// !!!!!!注意解决虚假唤醒,必须用while,不能用if
condition1.await();
}
//2.业务
System.out.println(Thread.currentThread().getName() + "=>" + number);
number = 2;
//3.通知
condition2.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
//-1
public void consumB() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number != 2) {
condition2.await();
}
//2.业务
System.out.println(Thread.currentThread().getName() + "=>" + number);
number = 3;
//3.通知
condition3.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
//+1
public void produceC() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number != 3) {// !!!!!!注意解决虚假唤醒,必须用while,不能用if
condition3.await();
}
//2.业务
System.out.println(Thread.currentThread().getName() + "=>" + number);
number = 4;
//3.通知
condition4.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
//-1
public void consumD() throws InterruptedException {
lock.lock();
try {
//1.判断等待
while (number != 4) {
condition4.await();
}
//2.业务
System.out.println(Thread.currentThread().getName() + "=>" + number);
number = 1;
//3.通知
condition1.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}
3 管程法