30天挑战的最后一天,我决定学习 Play Framework .我本来想写Sacla,但是研究几个小时后,我发现没法在一天内公正评价Scala,下个月花些时间来了解并分享经验。本文我们先来看看Play框架基础,再开发个程序。
Play 是一个开源的现代web框架,用Java和Scala写可扩展的web程序。它能自动加载更新使得极大提高生产率。Play设计了无状态,非阻塞的架构,这使得用Play框架开发水平扩展web程序很容易。
我学习Play的原因:
本文,我们来开发一个网摘程序,允许用户发布和分享链接,你可以在OpenShift上看 在线 程序。这和我们之前第22天开发的程序一样,你可以参考之前的用例来更好了解。
请参考 文档 了解怎样安装Play框架。
介绍完基础,我们来开始写程序。
在你机器上运行以下命令。
$ play new getbookmarks
_
_ __ | | __ _ _ _
| ‘_ \| |/ _‘ | || |
| __/|_|\____|\__ /
|_| |__/
play 2.2.1 built with Scala 2.10.2 (running Java 1.7.0_25), http://www.playframework.com
The new application will be created in /Users/shekhargulati/dev/challenges/30days30technologies/day30/blog/getbookmarks
What is the application name? [getbookmarks]
>
Which template do you want to use for this new application?
1 - Create a simple Scala application
2 - Create a simple Java application
> 2
OK, application getbookmarks is created.
Have fun!
如上,输入命令后,Play框架会问几个问题。首先是程序的名字,然后是我们是否想创建一个Scala程序或者Java程序。默认使用文件夹名作为程序名,我们选择第二个选项来创建Java程序。
以上命令新建了一个目录getbookmarks,生成一下文件和目录。
现在我们可以运行Play创建的默认程序,打开play控制台,运行play命令,然后用run命令。
$ cd getbookmarks
$ play
[info] Loading project definition from /Users/shekhargulati/dev/challenges/30days30technologies/day30/blog/getbookmarks/project
[info] Set current project to getbookmarks (in build file:/Users/shekhargulati/dev/challenges/30days30technologies/day30/blog/getbookmarks/)
_
_ __ | | __ _ _ _
| ‘_ \| |/ _‘ | || |
| __/|_|\____|\__ /
|_| |__/
play 2.2.1 built with Scala 2.10.2 (running Java 1.7.0_25), http://www.playframework.com
> Type "help play" or "license" for more information.
> Type "exit" or use Ctrl+D to leave this console.
[getbookmarks] $ run
[info] Updating {file:/Users/shekhargulati/dev/challenges/30days30technologies/day30/blog/getbookmarks/}getbookmarks...
[info] Resolving org.fusesource.jansi#jansi;1.4 ...
[info] Done updating.
--- (Running the application from SBT, auto-reloading is enabled) ---
[info] play - Listening for HTTP on /0:0:0:0:0:0:0:0:9000
(Server started, use Ctrl+D to stop and go back to the console...)
现在可以看到程序 http://localhost:9000 .
这个程序,我们只有一个域类, Story. 新建一个包模块然后新建Java类。
package models;
import play.db.ebean.Model;
import javax.persistence.Entity;
import javax.persistence.Id;
import java.util.Date;
@Entity
public class Story extends Model{
@Id
private String id;
private String url;
private String fullname;
private Date submittedOn = new Date();
private String title;
private String text;
private String image;
public Story() {
}
public Story(String url, String fullname) {
this.url = url;
this.fullname = fullname;
}
public Story(String url, String fullname, String image, String text, String title) {
this.url = url;
this.fullname = fullname;
this.title = title;
this.text = text;
this.image = image;
}
// Getter and Setter removed for brevity
}
以上代码定义了一个简单的JPA实体,用了@Entity和Id JPA注释,Play用它自己的ORM层Ebean, 每个实体类都扩展基础模块类。
Ebean默认是没激活的,要激活它,打开程序conf, 取消下面这行的注释。
ebean.default="models.*"
现在来激活程序的数据库,Play框架提供了内置支持的H2数据库,要激活它,打开程序conf文件,取消以下两行的注释。
db.default.driver=org.h2.Driver
db.default.url="jdbc:h2:mem:play"
现在刷新浏览器可以看到如下异常。
点击Apply this script now使SQL更新生效。
本文,我们开发了和第22天相同的程序,这个程序有AngularJS后端和REST后端,我们用Play框架写REST后端,重新用AngularJS后端,在conf/routes文件里,复制粘贴一下代码。
# Routes
# This file defines all application routes (Higher priority routes first)
# ~~~~
# Home page
GET / controllers.Assets.at(path="/public", file="/index.html")
GET /api/v1/stories controllers.StoryController.allStories()
POST /api/v1/stories controllers.StoryController.submitStory()
GET /api/v1/stories/:storyId controllers.StoryController.getStory(storyId)
# Map static resources from the /public folder to the /assets URL path
GET /assets/*file controllers.Assets.at(path="/public", file)
以上代码:
现在在控制器包里新建Java类,复制粘贴以下代码到StoryController.java 文件。
package controllers;
import com.fasterxml.jackson.databind.JsonNode;
import models.Story;
import play.api.libs.ws.Response;
import play.api.libs.ws.WS;
import play.db.ebean.Model;
import play.libs.Json;
import play.mvc.BodyParser;
import play.mvc.Controller;
import play.mvc.Result;
import play.mvc.Results;
import scala.concurrent.Await;
import scala.concurrent.Future;
import scala.concurrent.duration.Duration;
import java.util.List;
import java.util.concurrent.TimeUnit;
public class StoryController {
public static Result allStories(){
List<Story> stories = new Model.Finder<String , Story>(String.class, Story.class).all();
return Results.ok(Json.toJson(stories));
}
@BodyParser.Of(BodyParser.Json.class)
public static Result submitStory(){
JsonNode jsonNode = Controller.request().body().asJson();
String url = jsonNode.findPath("url").asText();
String fullname = jsonNode.findPath("fullname").asText();
JsonNode response = fetchInformation(url);
Story story = null;
if(response == null){
story = new Story(url,fullname);
}else{
String image = response.findPath("image").textValue();
String text = response.findPath("text").textValue();
String title = response.findPath("title").textValue();
story = new Story(url,fullname, image , text , title);
}
story.save();
return Results.created();
}
public static Result getStory(String storyId){
Story story = new Model.Finder<String, Story>(String.class, Story.class).byId(storyId);
if(story == null){
return Results.notFound("No story found with storyId " + storyId);
}
return Results.ok(Json.toJson(story));
}
private static JsonNode fetchInformation(String url){
String restServiceUrl = "http://gooseextractor-t20.rhcloud.com/api/v1/extract?url="+url;
Future<Response> future = WS.url(restServiceUrl).get();
try {
Response result = Await.result(future, Duration.apply(30, TimeUnit.SECONDS));
JsonNode jsonNode = Json.parse(result.json().toString());
return jsonNode;
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
}
以上代码:
我决定重新用第22天写的AngularJS前端,第22天展示了我们怎样用AngularJS和Java Spring框架,最好的部分是用JavaScript MV*框架,如果你的程序还是用REST接口客户端需求,你可以再用前端代码。详情参考第22天文章。
现在刷新浏览器访问程序 http://localhost:9000/
这就是今天的内容,希望你喜欢这个系列。
原文: https://www.openshift.com/blogs/day-30-play-framework-a-java-developer-dream-framework
[译] 第三十天:Play Framework - Java开发者梦寐以求的框架 - 百花宫
原文:http://www.cnblogs.com/zhiji6/p/4444797.html