我用阻塞队列编写了一个简单的消费者 - 生产者问题,该队列有多个生产者,多个消费者接受并将整数放在队列中。但是,当我尝试测试它时,结果并不理想,例如队列的大小不正确。我不认为消费者和生产者规模正在同步。此外,我对生产者和消费者都进行了2秒钟的睡眠,但在测试时,每两秒就会打印出2个生产者和2个消费者的结果。有谁知道我做错了什么?也许我开始错误的线程?我评论了另一种方式,但结果仍然是错误的。
结果:
run:
Producing 425 Thread-0 size left 0
Consuming 890 Thread-3 size left 0
Consuming 425 Thread-2 size left 0
Producing 890 Thread-1 size left 0
Consuming 192 Thread-2 size left 0
Consuming 155 Thread-3 size left 0
Producing 155 Thread-1 size left 0
Producing 192 Thread-0 size left 0
Consuming 141 Thread-2 size left 1
Producing 141 Thread-0 size left 0
Producing 919 Thread-1 size left 0
Consuming 919 Thread-3 size left 0
Producing 361 Thread-1 size left 0
Producing 518 Thread-0 size left 0
Consuming 518 Thread-3 size left 0
Consuming 361 Thread-2 size left 0
Producing 350 Thread-0 size left 1
Consuming 350 Thread-3 size left 0
Consuming 767 Thread-2 size left 0
Producing 767 Thread-1 size left 0
生产者
import java.util.Random;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Producer implements Runnable {
BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
public Producer(BlockingQueue<Integer> q) {
this.items = q;
}
private int generateRandomNumber(int start, int end) {
Random rand = new Random();
int number = start + rand.nextInt(end - start + 1);
return number;
}
public void run() {
for (int i = 0; i < 5; i++) {
int rand = generateRandomNumber(100, 1000);
try {
items.put(rand);
System.out.println("Producing " + rand + " " + Thread.currentThread().getName() + " size left " + items.size());
Thread.sleep(3000);
} catch (InterruptedException ex) {
Logger.getLogger(ProducerConsumer.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}
消费
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Consumer implements Runnable {
BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
public Consumer(BlockingQueue<Integer> q) {
this.items = q;
}
public void run() {
while (true) {
try {
System.out.println("Consuming " + items.take() + " " + Thread.currentThread().getName() + " size left " + items.size());
Thread.sleep(3000);
} catch (InterruptedException ex) {
Logger.getLogger(ProducerConsumer.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}
测试
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
public class ProducerConsumer {
public static void main(String args[]) {
BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
Producer producer = new Producer(items);
Consumer consumer = new Consumer(items);
Thread t1 = new Thread(producer);
Thread t2 = new Thread(producer);
Thread t3 = new Thread(consumer);
Thread t4 = new Thread(consumer);
/*
Thread t1 = new Thread(new Producer());
Thread t2 = new Thread(new Producer());
Thread t3 = new Thread(new Consumer());
Thread t4 = new Thread(new Consumer());
*/
t1.start();
t2.start();
t3.start();
t4.start();
}
}
更新:我尝试实现可重入锁,但我的程序在锁定行停止。有什么帮助吗? 消费者
import java.util.concurrent.locks.ReentrantReadWriteLock;
public class Consumer implements Runnable {
//private BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
private MyBlockingQ items;
public Consumer(MyBlockingQ q) {
this.items = q;
}
public void run() {
while (true) {
items.remove();
//Thread.sleep(1000);
}
}
}
生产者
import java.util.Random;
public class Producer implements Runnable {
private MyBlockingQ items;
public Producer(MyBlockingQ q) {
this.items = q;
}
private int generateRandomNumber(int start, int end) {
Random rand = new Random();
int number = start + rand.nextInt(end - start + 1);
return number;
}
public void run() {
for (int i = 0; i < 5; i++) {
int rand = generateRandomNumber(100, 1000);
items.add(rand);
}
}
}
MyBlockingQ(共享资源)
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.logging.Level;
import java.util.logging.Logger;
public class MyBlockingQ {
private BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
public MyBlockingQ() {
}
public void add(Integer i) {
lock.writeLock().lock();
try {
items.put(i);
System.out.println("Producing " + i + " " + Thread.currentThread().getName() + " size left " + items.size());
} catch (InterruptedException ex) {
Logger.getLogger(ProducerConsumer.class.getName()).log(Level.SEVERE, null, ex);
} finally {
lock.writeLock().unlock();
}
}
public void remove() {
lock.writeLock().lock();
try {
int taken = items.take();
System.out.println("Consuming " + taken + " " + Thread.currentThread().getName() + " size left " + items.size());
} catch (InterruptedException ex) {
Logger.getLogger(ProducerConsumer.class.getName()).log(Level.SEVERE, null, ex);
} finally {
lock.writeLock().unlock();
}
}
}
测试
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
public class ProducerConsumer {
public static void main(String args[]) {
MyBlockingQ items = new MyBlockingQ();
System.out.println("starting");
Thread t1 = new Thread(new Producer(items));
Thread t2 = new Thread(new Producer(items));
Thread t3 = new Thread(new Consumer(items));
Thread t4 = new Thread(new Consumer(items));
t1.start();
t2.start();
t3.start();
t4.start();
}
}
答案 0 :(得分:5)
这两行
items.put(rand);
System.out.println("Producing " + rand + " " + Thread.currentThread().getName() + " size left " + items.size());
未同步。生产者可能会将数字放入队列中,但是当从放入其中的线程显示队列的大小时,消费者可能已经消耗了一个数字。
答案 1 :(得分:3)
你可能会对这部分输出感到困惑:
Producing 425 Thread-0 size left 0
Consuming 890 Thread-3 size left 0
Consuming 425 Thread-2 size left 0
Producing 890 Thread-1 size left 0
问题:在Thread-1产生之前,Thread-3如何消耗890个项目?
答案: Thread-3在生成之前不会使用Thread-1消耗项目。
为什么:当Thread-1将项目放入队列时,Thread-3可能已经在等待从队列中获取的项目。所以Thread-1放置项目:
items.put(rand);
BEFORE Thread-1跳到下一行并打印有关它产生的项目的信息,Thread-3执行以下行:
System.out.println("Consuming " + items.take() + " " + Thread.currentThread().getName() + " size left " + items.size());
只有Thread-1执行其println:
System.out.println("Producing " + rand + " " + Thread.currentThread().getName() + " size left " + items.size());
因此,您可以在控制台中看到这些有趣的结果。
您可能想了解synchronizing。有两种方法可以解决您的问题:
同步锁定对同步块内对象的访问。这意味着每个其他方法必须等待轮到它才能访问对象。
因此,如果您对Producer和Consumer中的项目使用同步,那么:
我的情况是当商品为空且消费者的方法锁定商品时,程序将落入所谓的死锁。制作人必须等待消费者解锁,但它永远不会发生,因为消费者正在等待拍摄物品(必须由制作人放置)。
此外,我对生产者和消费者都进行了2秒的睡眠,但是 测试,每两秒打印出2个生产者的结果 和2个消费者。
这是你应该期待的。在Test类中,您将生产2个生产者和2个消费者。
Thread t1 = new Thread(producer);
Thread t2 = new Thread(producer);
Thread t3 = new Thread(consumer);
Thread t4 = new Thread(consumer);
t1.start();
t2.start();
t3.start();
t4.start();
答案 2 :(得分:0)
您需要同步items
访问权限。我只是略微改变了你的例子,结果看起来不错。由于同步,您还必须处理死锁。在这种情况下,只要您不在items.take()
的{{1}}上进行同步,它就应该没问题。
您的新测试:
Consumer
消费者
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
public class ProducerConsumer {
public static void main(String args[]) {
BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
Thread t1 = new Thread(new Producer(items));
Thread t2 = new Thread(new Producer(items));
Thread t3 = new Thread(new Consumer(items));
Thread t4 = new Thread(new Consumer(items));
t1.start();
t2.start();
t3.start();
t4.start();
}
}
制片人
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Consumer implements Runnable {
BlockingQueue<Integer> items = new LinkedBlockingQueue<>();
public Consumer(BlockingQueue<Integer> q) {
this.items = q;
}
public void run() {
while (true) {
try {
System.out.println("Consuming " + items.take() + " " + Thread.currentThread().getName() + " size left " + items.size());
Thread.sleep(1000);
} catch (InterruptedException ex) {
Logger.getLogger(ProducerConsumer.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}