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.
为什么会这样?我错过了什么?
因为wait()通过输入synchronized块释放它先前获得的锁定.从javadoc Object.wait:
线程释放此监视器的所有权并等待,直到另一个线程通过调用notify方法或notifyAll方法通知等待此对象监视器的线程唤醒.然后线程等待,直到它可以重新获得监视器的所有权并继续执行.
因此,您的第一个请求获取锁,输入doGet方法和调用wait(释放锁并等待).第二个请求获取锁定,进入doGet和调用notifyAll,这会唤醒第一个请求的线程.
在您使用它们之前仔细阅读文档以及/ 之前,或者您将遇到麻烦,这一点至关重要.waitnotifynotifyAll
| 归档时间: |
|
| 查看次数: |
1423 次 |
| 最近记录: |