1

我正在尝试使用类在并发 Java 中开发活动对象模式。java.util.concurrent

我用 aClient和 a来描述它Server。一个样本Server如下:

class Server implements Runnable {
  public final LinkedBlockingQueue que = new LinkedBlockingQueue();
  private final ExecutorService es = Executors.newCachedThreadPool();
  private Message currentMessage;

  private boolean state = false;

  public init() {
    es.submit(this);
  }

  public void requestForServer() {
    if (state) {
      this.currentMessage.await();
    }
    state = true;
  }

  public void run() {
     for(;;) {
       Message m = que.take();
       this.currentMessage = m;
       this.es.submit(m);           
     }
  }

}

和一个样本Client

class Client {

  private Server server;

  public Client(Server s) {
    this.server = s;
  }

  public void doSomething() {
    Message m = new Message(new Callable() {
      public Object call() {
        server.requestForServer();
      }
    });
    this.server.que.add(m);
  }

}

一个示例Message封装是:

class Message<V> extends FutureTask<V> {
  private Lock lock = new ReentrantLock();
  private Condition condition = new Condition();

  public Message(Callable<V> callable) {
    super(callable);
  }

  public void run() {
    try {
      lock.lock();
      super.run();
      lock.unlock();
    } catch(Exception e) {}
  }

  public void await() {
    try {
      condition.await();
    } catch(Exception e) {}
  }

  public void signal() {
    try {
      condition.signalAll();
    } catch(Exception e) {}
  }

}

和一个示例运行代码:

Server s = new Server();
Client c = new Client (s);
s.init();
c.doSomething();

我删除了一些实现细节来传达我的信息。

现在,问题是在什么时候,Server传入statetrue消息应该等待并且await在当前消息上调用。但是,我明白IllegalMonitorStateException这意味着当前消息不拥有当前线程来等待它。但是,我认为这很奇怪,因为当前消息在Server其线程池中被调用,因此当前消息也可以访问当前执行线程。

如果您有任何想法或建议,或者使用java.util.concurrent. 提前致谢。

更新:我在这篇
文中讨论了我可以部署的解决方案。我希望它可以帮助。

4

1 回答 1

3

当您等待相应的条件时,您必须实际获取锁。没有那个锁,你就不能直接将自己与条件联系起来。为了证明这一点:

  public void await() {
    lock.lock();
    try {
      condition.await();
    } catch(Exception e) {}
    finally{
      lock.unlock();
    }
  }

那应该解决您的 IllegalMonitorStateException

在正确性方面,您应该始终以 try{ } finally{ } 方式释放锁,您可以观察我作为示例编写的内容。这样做的原因是如果在两者之间发生异常lock().lock();并且super.run(); lock.unlock()永远不会被调用。

于 2011-03-02T16:16:37.987 回答