【发布时间】:2014-06-22 18:45:49
【问题描述】:
我编写了一个简单的 Java 程序来帮助使用线程和线程池来完成某些任务。在我的程序中,有 TheObject 类的对象必须对它们进行某种处理(在这种情况下,只是睡眠延迟并打印出它们的字段)。
TheObject 对象被放置在一个队列中,WorkerThreads 从中提取并处理它们。
我创建了一个Manager 类来初始化TheObjectss 和WorkerThreads。当我运行它时,我得到了奇怪的输出;不是线程关闭工作,而是一个线程处理所有事情!为什么会这样?如何让线程分担工作负载?
这是输出:
--------------------------------------------
alfred a 0
Thread-0
--------------------------------------------
bob b 1
Thread-0
--------------------------------------------
carl c 2
Thread-0
--------------------------------------------
dave d 3
Thread-0
--------------------------------------------
earl e 4
Thread-0
--------------------------------------------
fred f 5
Thread-0
--------------------------------------------
greg g 6
Thread-0
--------------------------------------------
harry h 7
Thread-0
--------------------------------------------
izzie i 8
Thread-0
--------------------------------------------
jim j 9
Thread-0
--------------------------------------------
kyle k 0
Thread-0
--------------------------------------------
Larry L 1
Thread-1
--------------------------------------------
Michael M 2
Thread-1
--------------------------------------------
Ned N 3
Thread-0
--------------------------------------------
Olaf O 4
Thread-0
--------------------------------------------
Peter P 5
Thread-0
--------------------------------------------
Quincy Q 6
Thread-0
--------------------------------------------
Raphael R 7
Thread-0
--------------------------------------------
Sam S 8
Thread-0
--------------------------------------------
Trixie T 9
Thread-0
代码:
类:
- 经理
- 对象
- 对象队列
- 工作线程
经理
public class Manager {
public static void main(String[] args) throws InterruptedException {
TheObjectQueue queue = new TheObjectQueue();
//Create Objects
int numberOfInitialObjects = 10;
String[] arrayOfNames = {"alfred", "bob", "carl", "dave", "earl", "fred", "greg", "harry", "izzie", "jim"};
//char[] arrayOfChars = {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'};
//int[] arrayOfNums = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
for (int i = 0; i < numberOfInitialObjects; i++){
TheObject anObject = new TheObject(arrayOfNames[i], arrayOfNames[i].charAt(0), i);
queue.addToQueue(anObject);
}
int numberOfThreads = 2;
for (int i = 0; i < numberOfThreads; i++){
WorkerThread workerThread = new WorkerThread(queue);
workerThread.start();
}
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
String[] arrayOfNames2 = {"kyle", "Larry", "Michael", "Ned", "Olaf", "Peter", "Quincy", "Raphael", "Sam", "Trixie"};
for (int i = 0; i < numberOfInitialObjects; i++){
TheObject anObject = new TheObject(arrayOfNames2[i], arrayOfNames2[i].charAt(0), i);
queue.addToQueue(anObject);
}
}
}
对象
public class TheObject {
private String someName;
private char someChar;
private int someNum;
public TheObject(String someName, char someChar, int someNum) {
super();
this.someName = someName;
this.someChar = someChar;
this.someNum = someNum;
}
public String getSomeName() {
return someName;
}
public char getSomeChar() {
return someChar;
}
public int getSomeNum() {
return someNum;
}
}
对象队列
import java.util.LinkedList;
public class TheObjectQueue {
private LinkedList<TheObject> objectQueue = null;
public TheObjectQueue(){
objectQueue = new LinkedList<TheObject>();
}
public LinkedList<TheObject> getQueue(){
return objectQueue;
}
public void addToQueue(TheObject obj){
synchronized (this) {
objectQueue.addFirst(obj);
this.notify();
}
}
public TheObject removeFromQueue(){
synchronized (this) {
TheObject obj = objectQueue.removeLast();
return obj;
}
}
public int getSize(){
return objectQueue.size();
}
public boolean isEmpty(){
return objectQueue.isEmpty();
}
}
工作线程
import java.util.Random;
public class WorkerThread extends Thread{
private TheObjectQueue queue;
public WorkerThread(TheObjectQueue queue){
this.queue = queue;
}
public void process(TheObject obj){
//Stall for a random amount of time
Random random = new Random();
int randomInt = random.nextInt(3)*1000;
try {
Thread.sleep(randomInt);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//Begin Printing
System.out.println("--------------------------------------------");
System.out.print(obj.getSomeName());
System.out.print(" ");
System.out.print(obj.getSomeChar());
System.out.print(" ");
System.out.println(obj.getSomeNum());
System.out.println(this.getName());
}
@Override
public void run() {
while(true){
synchronized (queue) {
while(queue.isEmpty()){
try {
queue.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
TheObject objToDealWith = queue.removeFromQueue();
process(objToDealWith);
super.run();
}
}
}
}
【问题讨论】:
-
为什么不使用 ExecutorService?
-
@Jarrod 我不明白这是怎么复制的。即使存在以不同方式执行此操作的工具,问题也不在于它们的使用。我要重新营业了。
-
此行为的一个可能原因是糟糕,不,你调用isEmpty()未同步,因此所有其他工作线程(thread-0除外)总是将队列视为空。isEmpty锁定queue,所以isEmpty()应该按预期工作。
标签: java multithreading synchronization threadpool