gpt4 book ai didi

java - 为什么这个 ReadWriteLock 示例不起作用?

转载 作者:行者123 更新时间:2023-11-30 07:32:22 26 4
gpt4 key购买 nike

LinkedList 在尝试轮询数据时抛出异常。但我认为我正确地使用了读/写锁的概念。该代码有什么问题?

package sample;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;


public class PingPong extends Thread {
boolean read = false;
Queue<String> queue;
static ReadWriteLock lock = new ReentrantReadWriteLock();
final static Lock readLock = lock.readLock();
final static Lock writeLock = lock.writeLock();
boolean stop;

public PingPong(boolean read, Queue<String> queue) {
this.read = read;
this.queue = queue;
}

int count = 0;

@Override
public String toString() {
return "PingPong{" +
"read=" + read +
", count=" + count +
'}';
}

@Override
public void run() {
if (read) {
while (!stop) {

readLock.lock();
// synchronized (queue) {
try {

String string = queue.poll();
if (string != null) {
count++;
}
} finally {
readLock.unlock();
}


// }
inform();
}
} else {
while (!stop) {

writeLock.lock();
// synchronized (queue) {
try {
if (queue.add("some str" + count)) {
count++;
}
} finally {
writeLock.unlock();
}

// }

inform();
}

}


}

private void inform() {
// Thread.yield();
// synchronized (queue) {
// queue.notify();
// try {
// queue.wait(1);
// } catch (InterruptedException e) {
// e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates.
// }
// }
}

public static void main(String[] args) throws InterruptedException {
Queue<String> queue = new LinkedList();
// queue = new ArrayBlockingQueue<String>(100);
// queue = new ConcurrentLinkedQueue<String>();
List<PingPong> pongs = new ArrayList<PingPong>();
for (int i = 0; i < 10; ++i) {
PingPong pingPong = new PingPong(i % 2 == 0, queue);
pingPong.start();
pongs.add(pingPong);
}
Thread.sleep(1000);
int sum = 0;
int read = 0;
int write = 0;
for (PingPong pp : pongs) {
pp.stop = true;
pp.join();
}
for (PingPong pp : pongs) {
System.out.println(pp);
sum += pp.count;
if (pp.read) read += pp.count;
else write += pp.count;
}
System.out.println(sum);
System.out.println("write = " + write);
System.out.println("read = " + read);
System.out.println("queue.size() = " + queue.size());
System.out.println("balance (must be zero) = " + (write - read - queue.size()));

}
}

最佳答案

这是因为这个调用改变了 queue 集合:

String string = queue.poll();

来自 Queue Java文档:

Retrieves and removes the head of this queue, or returns null if this queue is empty.

读取锁旨在用于多个线程可以安全地读取,而写入必须以独占方式执行(没有其他读取和写入)的情况。因为您正在使用读锁来轮询队列(写操作!),所以您实际上允许多个线程同时修改非线程安全的LinkedList

在这种情况下,读写锁不是正确的同步机制。

关于java - 为什么这个 ReadWriteLock 示例不起作用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6643995/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com