本文的目标是经过一个用户登陆示例,简要的介绍使用Play进行Web开发的基本流程。本文并不会手把手教你如何建立一个Play应用,而是经过核心的代码片断传递Play的一些设计理念,为不熟悉Play框架的同窗提供一个快速了解的途径。html
在controllers
目录下建立ApplicationController
类:api
package controllers import play.api.mvc._ class ApplicationController extends Controller { def login = Action { Ok(views.html.login("用户登陆")) } def doLogin(userName: String, password: String) = Action { val mess = userName + "&" + password Ok(mess) } }
上面定义了login
和doLogin
两个Action,一个用于引导用户至登陆页面,另外一个用户处理登陆请求。一个Action其实就是一个函数,接受一个request做为参数,返回一个Result,返回的Result最终会被以Http响应的形式写回给浏览器。Ok(mess)
返回的结果就是Result类型。浏览器
不熟悉Scala的同窗看上面的代码会感受比较奇怪,
Action{...}
和Ok(...)
是什么鬼?其实这是调用单例对象上apply
方法的简写形式,即Action{...}
等价于Action.apply(...)
,Ok(...)
等价于Ok.apply(...)
。省略掉.apply
是否是看起来感受舒服一点^_^。 另外Scala不建议使用return
语句,默认最后一条语句的值做为函数的返回值。多线程
在views
目录下建立login.scala.html
:mvc
@(title: String) <!DOCTYPE html> <html lang="en"> <head><title>@title</title></head> <body> <form action="/doLogin"> <input name="userName" type="text" placeholder="User Name" /> <input name="password" type="password" placeholder="Password" /> <button type="submit">当即登陆</button> </form> </body> </html>
咱们知道模板页面中有两部份内容,一部分是不可变的Html内容,另外一部分是须要动态执行的代码。而神奇的@
符号就是要告诉Play,它后面跟着的是须要动态执行的代码。在Play中,一个模板文件就是一个函数,接受一组参数,返回动态执行后的Html内容,函数名就是不带后缀的文件名,例如上面定义的模板文件编译后生成的函数名称是login
。模板文件的第一行用于指明函数的参数列表,上面的模板文件至关于定义了一个login(title: String)
函数。app
把View抽象成函数好处仍是不少的,例如组合多个View变成了函数之间的相互调用,另外咱们也可使用多线程加速大页面的渲染。 Play的模板层采用
Scala
语言编写,借助Scala语言,在Play的模板层你会感受本身像是一只脱了缰的野马。其实在模板层只须要了解Scala的if
和for
语法便可。Scala虽然入门门槛较高,可是带来的收益是巨大的,随着你对Play了解的深刻必定能够慢慢的体会到这点。框架
Play使用routes文件定义Http请求和Action之间的映射关系,编辑conf/routes
文件,添加一行:函数
GET /login controllers.ApplicationController.login GET /doLogin controllers.ApplicationController.doLogin(userName: String, password: String)
进入命令行,执行activator run
,在浏览器中打开http://localhost:9000/login
:
单元测试
一般登陆操做使用Post请求,因此咱们调整一下routes:测试
POST /doLogin controllers.ApplicationController.doLogin
ApplicationController
代码调整以下:
package controllers import play.api.mvc._ import play.api.data._ import play.api.data.Forms._ class ApplicationController extends Controller { def login = Action { Ok(views.html.login("用户登陆")) } def doLogin = Action { implicit request => val loginForm = Form( tuple( "userName" -> email, "password" -> text(minLength = 6) ) ) loginForm.bindFromRequest().fold( errorForm => Ok(errorForm.errors.toString()), tupleData => { val (userName, password) = tupleData Ok(userName + "&" + password) } ) } }
上面的fold
函数用于处理表单参数验证经过和不经过两种状况。
经过上面简单的登陆示例咱们会发现,Play中Controller和View是两个独立的模块,之间没有任何耦合。Controller完成一些业务运算,而后将数据以参数的形式传递给View,View没有任何的内置对象,全部的依赖都定义在参数列表中,Controller和View之间只是简单的函数调用关系,状态经过函数参数进行传递,这样的好处是程序执行流程容易追踪,代码容易阅读,而且单元测试会变得很是简单,固然最大的好处是多线程环境下代码无需同步,极大地提升了执行效率。另外Play在改动代码后无需重启,直接刷新浏览器就能够了,开发体验仍是不错的。