Java中的线程:如何锁定对象?

2022-09-01 08:02:16

以下函数在其自己的线程中执行:

private void doSendData()
{
    try {

           //writeToFile(); // just a temporary location of a call
           InetAddress serverAddr = InetAddress.getByName(serverAddress);
           serverAddr.wait(60000);
           //Log.d("TCP", "C: Connecting...");
           Socket socket = new Socket(serverAddr, portNumber);
           socket.setSoTimeout(3000);

               try {
                //Log.d("TCP", "C: Sending: '" + message + "'");
                PrintWriter out = new PrintWriter( new BufferedWriter( new OutputStreamWriter(socket.getOutputStream())),true);
                String message = packData();
                out.println(message);
                Log.d("TCP", "C: Sent.");
                Log.d("TCP", "C: Done.");
                connectionAvailable = true;

             } catch(Exception e) {
                 Log.e("TCP", "S: Error", e);
                 connectionAvailable = false;

               } finally {
                  socket.close();
                  announceNetworkAvailability(connectionAvailable);
                }

         } catch (Exception e) {
              Log.e("TCP", "C: Error", e);
              announceNetworkAvailability(connectionAvailable);
         }
}

当执行到达该行时,它会引发异常:serverAddr.wait(60000)

java.lang.IllegalMonitorStateException: object not locked by thread before wait()

有谁知道如何锁定对象或函数以防止并发?我尝试添加一个 Lock 对象:

private final Lock lock = new ReentrantLock();

和行

boolean locked = lock.tryLock();

在功能开始时,但它不起作用。


答案 1

为了在对象上调用 wait(),您必须在该对象上保持同步锁(尽管该锁实际上是在线程等待时释放的):

synchronized (serverAddr) {
  serverAddr.wait();
}

我不得不承认,在这种情况下,你为什么要这样做让我感到困惑......


答案 2

也许你正在寻找的方法是Thread.sleep(long)?此方法将在恢复之前等待(如停止线程的执行)指定的时间(以毫秒为单位)。

object.wait(long) (这是你正在使用的)做了一些完全不同的事情。它等待来自另一个线程的另一个对象通知它(即:向它发送一种唤醒消息),并且最多等待指定的毫秒数。鉴于您发布的代码,我非常怀疑这是您真正想要的。

如果 Thread.sleep() 不是你想要的,那么你应该使用其他海报提到的同步块。


推荐