源代码下载地址:http://download.csdn.net/source/1662193
一. 认识REST
REST软件架构是由Roy Thomas Fielding博士在2000年首次提出的。他为我们描绘了开发基于互联网的网络软件的蓝图。REST软件架构是一个抽象的概念,是一种为了实现这一互联网的超媒体分布式系统的行动指南。利用不论什么的技术都能够实现这样的理念。而实现这一软件架构最著名的就是HTTP协议。通常我们把REST也写作为REST/HTTP,在实际中往往把REST理解为基于HTTP的REST软件架构,或者更进一步把REST和HTTP看作为等同的概念。
很多其它请阅读:
http://www.infoq.com/cn/articles/rest-architecure
http://www.kuqin.com/system-analysis/20080515/8518.html
REST 的请求流程示意图
个人总结:
1. REST架构仅仅是一种思想,并没有限定不论什么技术,不论什么语言。
2. REST的本质就是HTTP调用,用于减少各个应用之间的耦合度。
3. 良好的REST 架构,应该有统一的表示方式和数据格式,可以有效的把各种资源组织起来,并可以进行有效的控制。
二. 实现REST架构
1.框架设计
2.接口的定义
IRestRequest : 用来表示 REST请求
IRestResponse:用来表示 REST响应
IRestInterceptor:用来表示REST拦截器
RestException: 用来表示REST异常
3.实现的主要代码
初始化代码,借助Servlet的init
@Override
public void init(ServletConfig config) throws ServletException {
//1.从init里载入对应的Rest服务,假设是有Spring框架或者别的,原理都一样
String serviceClass = config.getInitParameter("service-class");
if (serviceClass != null) {
System.out.println("Rest服务:" + serviceClass);
String[] classes = serviceClass.split(",");
try {
for (String className : classes) {
Class newClass = Class.forName(className);
Object newObject = newClass.newInstance();
if (newObject instanceof IRestService) {
IRestService restService = (IRestService) newObject;
services.put(restService.getURI(), restService);
System.out.println("载入Rest服务:" + newObject.getClass().getName() + ",URI=" + restService.getURI());
}
}
} catch (Exception e) {
System.out.println("载入Rest服务出错:" + e.getMessage());
}
}
//2.载入拦截器
String interceptorClas = config.getInitParameter("interceptor-class");
if (interceptorClas != null) {
System.out.println("拦截器:" + serviceClass);
String[] classes = interceptorClas.split(",");
try {
for (String className : classes) {
Class newClass = Class.forName(className);
Object newObject = newClass.newInstance();
if (newObject instanceof IRestInterceptor) {
IRestInterceptor interceptor = (IRestInterceptor) newObject;
interceptors.add(interceptor);
System.out.println("载入Rest拦截器:" + newObject.getClass().getName());
}
}
} catch (Exception e) {
System.out.println("载入Rest拦截器出错:" + e.getMessage());
}
}
转发实现
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//处理HTTP请求
response.setContentType("text/html;charset=GBK");
response.setCharacterEncoding("GBK");
Response returnResponse = new Response(); //终于返回的结果,能够是JSON或者XML格式
try {
//1.获得请求的URI
String uri = request.getRequestURI();
//System.out.println("URI===="+uri);
//2.获得对应的RestService
IRestService service = services.get(uri);
if (service != null) {
//2.0 校验service是否符合当前环境
//--------------
//2.1 构造对应的 request 和 response 上下文
GenericRestRequest restRequest = new GenericRestRequest(request);
GenericRestResponse restResponse = new GenericRestResponse(response);
//2.2 填充环境变量之类
restRequest.setRestService(service);
restResponse.setRestService(service);
//2.3 运行拦截器
for (IRestInterceptor interceptor : interceptors) {
interceptor.handleRest(restRequest, restResponse);
}
//2.4 运行服务
service.service(restRequest, restResponse);
//2.5
if (restResponse.getResponseData() != null) {
returnResponse = restResponse.getResponseData();
}
} else {
throw new Exception("未找到相应的Rest服务:" + uri);
}
} catch (Exception e) {
e.printStackTrace();
returnResponse.addError("doAction", e.getMessage());
}
response.getWriter().write(returnResponse.toJSON());
}
原型源代码下载:http://download.csdn.net/source/1662193
三. 数据格式
先看看曾经写的一篇文章: http://blog.csdn.net/maoxiang/archive/2008/06/25/2584282.aspx 《改善Form提交数据的 UI 交互设计》
数据格式定义例如以下:
JSON 格式:
{
code: 200|302|403|500 , 200表示正常,302表示跳转,403表示须要验证码,500异常
messages:{ //传递的数据
[key:value]
}
XML格式:
<response>
<code>200|302|403|500</code>
<messages>
<key></key>
<value></value>
</messages>
</response>
举例说明:
{"code":200,"error":false,"messages":{"list":[{"name":"user0"},{"name":"user1"},{"name":"user2"},{"name":"user3"},{"name":"user4"}]},"ok":true,"redirect":false,"verify":false,"version":2}
採用javascript 来解析JSON格式就相对简单非常多了:
if (data.code==200){ //假设是成功返回
var users= data.messages.list; //这个由rest服务返回
var html="以ol方式显示数据:<br/><ol>”
for(var i=0;i<users.length;i++){
html+="<li>"+users[i].name+"</li>";
}
html+="</ol>";
$("#idResult").html(html);
}else{
//出错了,或者别的
alert(data.messages.doAction);
}
四. 实际应用
1. 太平洋女性网搜店系统 http://shop.pclady.com.cn
2. 太平洋女性网晒客系统 http://blog.pclady.com.cn
3. 太平洋女性网化妆品库 http://cosme.pclady.com.cn
.....
简单的REST的框架实现