似乎从同一会话同时调用同一个端点会在Play端触发一些问题。
虽然请求具有不同的参数,但有时会从先前的请求中复制(缓存?)这些参数。
这是我如何设置测试来重现这个:
路线:
GET /api/test @com.standup75.TestController.test(test: String ?= null)
GET /api/testPage @com.standup75.TestController.testPage()
TestController.java
....
public Result test(String test) {
return ok("test = " + test);
}
public Result testPage() {
return ok(views.test.render());
}
test.html.scala
@()
<h1>Test page</h1>
<ul id="log"></ul>
<script src="//code.jquery.com/jquery-1.10.1.min.js"></script>
<script>
$(function(){
var $ul = $("#log");
for (var i = 0; i < 20; i++) {
(function(i){
var url = "/api/test?test="+i;
$.ajax({
url: url,
dataType: 'html',
success: function(res){
$ul.append("<li>url: " + url + " - result: " + res + "</li>");
}
});
})(i)
}
});
</script>
当我转到/ api / testPage时,这是令人惊讶的输出:
Test page
url: /api/test?test=3 - result: test = 4
url: /api/test?test=5 - result: test = 2
url: /api/test?test=0 - result: test = 2
url: /api/test?test=1 - result: test = 4
url: /api/test?test=4 - result: test = 4
url: /api/test?test=2 - result: test = 3
url: /api/test?test=7 - result: test = 6
url: /api/test?test=8 - result: test = 8
url: /api/test?test=6 - result: test = 6
url: /api/test?test=9 - result: test = 10
url: /api/test?test=10 - result: test = 10
url: /api/test?test=11 - result: test = 11
url: /api/test?test=13 - result: test = 13
url: /api/test?test=14 - result: test = 14
url: /api/test?test=12 - result: test = 12
url: /api/test?test=17 - result: test = 16
url: /api/test?test=15 - result: test = 16
url: /api/test?test=16 - result: test = 16
url: /api/test?test=18 - result: test = 19
url: /api/test?test=19 - result: test = 19
结果中的测试值并不总是与测试参数相同,这对我没有意义?
答案 0 :(得分:2)
前一段时间我使用Spring bean作为控制器时遇到了同样的问题。
我看到你正在使用控制器实例,而不是静态方法(来自@)。你有机会使用play.mvc.Security.AuthenticatedAction吗?你如何创建它的实例?它需要是每个请求的新实例(Spring bean案例中的原型范围),否则play会以非确定的方式混合请求。
请参阅https://groups.google.com/forum/#!topic/play-framework/EHl9mbafUlA
答案 1 :(得分:1)
我与OP合作,我们最终弄明白了。这个问题与frant.hartm提到的类似。
控制器继承了一个基类,该基类在请求委托链中为使用@With的所有请求添加了许多Play动作。一些Action类型也是Spring组件,默认为单例。当Play初始化请求的委托链时,这些Actions的实例最终可能会被多个同时提供的请求共享,并且设置其委托链的最后一个请求将设置root操作。基本上,共享Action的“delegate”成员的竞争条件。
通过将@Scope(“prototype”)添加到相关的Actions来解决问题。感谢frant的提示。