从同一会话同时请求同一端点会在Play中产生问题

时间:2014-02-07 16:53:12

标签: java concurrency playframework playframework-2.0

似乎从同一会话同时调用同一个端点会在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

结果中的测试值并不总是与测试参数相同,这对我没有意义?

2 个答案:

答案 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的提示。