Servlet doGet同步 - 不起作用?

slo*_*mir 4 java multithreading servlets synchronized

我知道这是一个简单的问题,但我有些困惑.

如果我理解得很好,简单来说,当请求到达Web服务器时,他会为每个请求创建一个线程给某个servlet.

考虑到我们在MyServlet中有下一个代码(我遗漏了异常处理和类似的代码):

synchronized protected void doGet( ... ...){
    PrintWritet pw=response.getWriter();
    String param=request.getParameter("p");

    if(param.equals("a")){
        wait();
    }else{
        notifyAll();
    }

    pw.write("Hello!");
}
Run Code Online (Sandbox Code Playgroud)

我希望这个servlet会卡住,因为进入这个方法的第一个线程(带有param = a)将永远等待,因为任何其他未来的线程都会因为synchronized关键字而停留在doGet前面,并且因为那个notifyAll将永远不会得到执行.

现在,如果我在浏览器中打开新选项卡并点击/ MyServlet?p = a,浏览器等待127.0.0.1 ...之后,我打开新标签并点击/ MyServlet?p = b(或任何东西!= a )第一个标签发布并打印出"你好!" 信息.

这意味着第二个线程已进入doGet,并执行notifyAll.

为什么会这样?我错过了什么?

ska*_*man 8

因为wait()通过输入synchronized块释放它先前获得的锁定.从javadoc Object.wait:

线程释放此监视器的所有权并等待,直到另一个线程通过调用notify方法或notifyAll方法通知等待此对象监视器的线程唤醒.然后线程等待,直到它可以重新获得监视器的所有权并继续执行.

因此,您的第一个请求获取锁,输入doGet方法和调用wait(释放锁并等待).第二个请求获取锁定,进入doGet和调用notifyAll,这会唤醒第一个请求的线程.

在您使用它们之前仔细阅读文档以及/ 之前,或者您将遇到麻烦,这一点至关重要.waitnotifynotifyAll