Actions, Controllers and Results

    play.api.mvc.Action 就是一个处理收到的请求然后产生结果发给客户端的函数(play.api.mvc.Request => play.api.mvc.Result)。

    Action 返回一个类型为 play.api.mvc.Result 的值,代表了发送到 web 客户端的 HTTP 响应。在这个例子中,OK 构造了一个 200 OK 的响应,并包含一个 text/plain 类型的响应体。

    构建一个 Action

    play.api.mvc.Action 的伴生对象(companion object)提供了一些 helper 方法来构造一个 Action 值。

    最简单的一个函数是 OK,它接受一个表达式块作为参数并返回一个 Result

    1. Action {
    2. Ok("Hello world")
    3. }

    这是构造 Action 的最简单方法,但在这种方法里,我们并没有使用传进来的请求。实际应用中,我们常常要使用调用这个 Action 的 HTTP 请求。

    因此,还有另一种 Action 构造器,它接受一个函数 Request => Result 作为输入:

    1. Action { request =>
    2. Ok("Got request [" + request + "]")
    3. }
    1. Action { implicit request =>
    2. Ok("Got request [" + request + "]")
    3. }

    最后一种创建 Action 的方法是指定一个额外的 BodyParser 参数:

    这份手册后面会讲到 Body 解析器(Body Parser)。现在你只需要知道,上面讲到的其它构造 Action 的方法使用的是一个默认的解析器:任意内容 body 解析器(Any content body parser)。

    一个 就是一个产生 Action 值的单例对象。

    定义一个 action 生成器的最简单方式就是定义一个无参方法,让它返回一个 Action 值:

    1. import play.api.mvc._
    2. object Application extends Controller {
    3. def index = Action {
    4. Ok("It works!")
    5. }
    6. }

    当然,生成 action 的方法也可以带参数,并且这些参数可以在 Action 闭包中访问到:

    1. def hello(name: String) = Action {
    2. Ok("Hello " + name)
    3. }

    简单结果

    到目前为止,我们就只对简单结果感兴趣:HTTP 结果。它包含了一个状态码,一组 HTTP 报头和发送给 web 客户端的 body。

    1. def index = Action {
    2. Result(
    3. body = Enumerator("Hello world!".getBytes())
    4. )
    5. }

    当然,Play 提供了一些 helper 方法来构造常见结果,比如说 OK。下面的代码和上面的代码是等效的:

    上面两段代码产生的结果是一样的。

    下面是生成不同结果的一些例子:

    1. val ok = Ok("Hello world!")
    2. val notFound = NotFound
    3. val pageNotFound = NotFound(<h1>Page not found</h1>)
    4. val badRequest = BadRequest(views.html.form(formWithErrors))
    5. val oops = InternalServerError("Oops")
    6. val anyStatus = Status(488)("Strange response type")

    上面的 helper 方法都可以在 play.api.mvc.Results 特性(trait)和伴生对象(companion object)中找到。

    重定向到一个新的 URL 是另一种简单结果。然而这些结果类型并不包含一个响应体。

    同样地,有一些 helper 方法可以来创建重定向结果:

    1. def index = Action {
    2. Redirect("/user/home")
    3. }
    1. def index = Action {
    2. Redirect("/user/home", MOVED_PERMANENTLY)
    3. }

    「TODO」 dummy 页面

    你可以使用一个定义为 TODO 的空的 Action 实现,它的结果是一个标准的 页面: