Spring 4.2.4.RELEASE MVC 学习笔记 - 7 - RestFull API(咋个办呢 zgbn)
来源:互联网 发布:http post web数据 编辑:程序博客网 时间:2024/06/08 17:52
Spring 4.2.4.RELEASE MVC 学习笔记 - 7 - RestFull API
本小节主要学习一下,如何整合spring mvc gson基于框架本身的发布restfull api。
1、看一下pom.xml
由于之前我们引用过Gson的依赖jar,所以这次并不需要修改什么配置。
<?xml version="1.0"?><project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> <modelVersion>4.0.0</modelVersion> <parent> <!-- 因为是 _total项目的子项目 --> <groupId>cn.vfire.frameword</groupId> <artifactId>total</artifactId> <version>0.0.1-SNAPSHOT</version> </parent> <artifactId>framework_spring</artifactId> <packaging>war</packaging> <name>framework_spring Maven Webapp</name> <url>http://maven.apache.org</url> <properties> <org_springframework_version>4.2.4.RELEASE</org_springframework_version> </properties> <dependencies> <!-- 导入junit jar包 --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <!-- 导入@Getter @Setter自动编码工具jar --> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.16.6</version> </dependency> <!-- 导入jsp servlet规范jar包 --> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>3.0.1</version> </dependency> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>jsp-api</artifactId> <version>2.2</version> </dependency> <!-- 导入spring框架依赖jar包 --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>${org_springframework_version}</version> </dependency> <dependency> <!-- 该jar用于spring 支持 junit测试 --> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${org_springframework_version}</version> </dependency> <!-- spring-context-support,该jar引入能使spring mvc非常好的对freemark的支持 --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context-support</artifactId> <version>${org_springframework_version}</version> </dependency> <!-- 导入freemarker视图解析框架jar与sprig集成 --> <dependency> <groupId>org.freemarker</groupId> <artifactId>freemarker</artifactId> <version>2.3.22</version> </dependency> <!-- 导入log4j的jar包 --> <dependency> <groupId>log4j</groupId> <artifactId>log4j</artifactId> <version>1.2.17</version> </dependency> <!-- 导入Gson json工具包 --> <dependency> <groupId>com.google.code.gson</groupId> <artifactId>gson</artifactId> <version>2.6</version> </dependency> <!-- _total的引用子项目 --> <dependency> <groupId>cn.vfire.frameword</groupId> <artifactId>framework_common</artifactId> <version>${project.parent.version}</version> </dependency> </dependencies> <build> <finalName>framework_spring</finalName> <!-- 修改maven编译输出目录 --> <outputDirectory>src/main/webapp/WEB-INF/classes</outputDirectory> <testOutputDirectory>src/main/webapp/WEB-INF/classes</testOutputDirectory> <plugins> <!-- 添加一个mavne的插件,作用是在我通过maven发布的时候,能将依赖的jar复制一份到我指定的目录下。(个人比较懒) --> <plugin> <artifactId>maven-antrun-plugin</artifactId> <executions> <execution> <id>copy-lib-src-webapps</id> <phase>package</phase> <configuration> <tasks> <delete dir="src/main/webapp/WEB-INF/lib" /> <copy todir="src/main/webapp/WEB-INF/lib"> <fileset dir="${project.build.directory}\${project.build.finalName}\WEB-INF\lib"> <include name="*" /> </fileset> </copy> </tasks> </configuration> <goals> <goal>run</goal> </goals> </execution> </executions> </plugin> </plugins> </build></project>
注意spring-mvc-servlet.xml变化
spring-mvc-servlet.xml配置文件中,在mvc:annotation-driven元素中我注入一个GSONHttpMessageConverte类为Gson数据解析器,用于对Controller方法中返回的对象进行json数据解析。
记性好的同学,看到这里有没有仿佛见过的印象,是不是在前小结我们解决中文乱码的时候再也同样的位置注入一个spring mvc框架提供的StringHttpMessageConverter解析器用于解析字符串类型数据的。
起始简单理解,spring mvc在处理request、response的数据的时候,是通过不同的数据解析器来完成的。所以这里我们注入一个用Gson来处理request、response传递的数据Json化。
/framework_spring/src/main/resources/config/spring/spring-mvc-servlet.xml
<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.2.xsd"> <!-- 开启spring 上下文注解支持 --> <context:annotation-config /> <!-- 把标记了@Controller注解的类转换为bean --> <context:component-scan base-package="cn.vfire.framework" /> <!-- 开启spring mvc注解支持 --> <mvc:annotation-driven> <mvc:message-converters register-defaults="true"> <bean class="cn.vfire.framework.spring.mvc.converter.GSONHttpMessageConverte"> <property name="excludeFieldsWithoutExposeAnnotation" value="false" /> </bean> </mvc:message-converters> </mvc:annotation-driven> <!-- 如果当前请求为“/”时,则转发到“/helloworld/index” --> <mvc:view-controller path="/" view-name="forward:/index.jsp" /> <!-- 设置默认的Servlet来响应静态文件 --> <mvc:resources mapping="/resource" location="/resource" /> <!-- 当上面要访问的静态资源不包括在上面的配置中时,则根据此配置来访问 --> <mvc:default-servlet-handler /> <!-- 启动Spring MVC的注解功能,完成请求和注解POJO的映射 --> <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" /></beans>
构造GSONHttpMessageConverte解析器
这个地方我就直接上代码了,这里需要强调一点,自定义的GSONHttpMessageConverte需要实现Spring mvc 提供的 AbstractHttpMessageConverter接口,由于Controller方法返回的树Mode对象,所以泛型我直接指定Object。
PS:别问我为什么我就知道这么写,请大家没事阅读一下Spring mvc提供的StringHttpMessageConverter解析器的源代码,基础好的同学照着写都可以写出来的,很好理解。
模仿:org.springframework.http.converter.StringHttpMessageConverter
在下面GSONHttpMessageConverte中,我预留出来四个成员属性
availableCharsets、excludeFieldsWithoutExposeAnnotation、dateFormat、writeAcceptCharset,
这几个成员属性都有对应的Setter方法,也就是说在注入GSONHttpMessageConverte对象的时候,可以直接利用spring输入属性值的方式修改默认值。(别问我这几个都是干啥的,自己读代码。)
cn.vfire.framework.spring.mvc.converter.GSONHttpMessageConverte
package cn.vfire.framework.spring.mvc.converter;import java.io.IOException;import java.lang.reflect.Type;import java.nio.charset.Charset;import java.util.ArrayList;import java.util.List;import lombok.Getter;import lombok.Setter;import org.springframework.http.HttpInputMessage;import org.springframework.http.HttpOutputMessage;import org.springframework.http.MediaType;import org.springframework.http.converter.AbstractHttpMessageConverter;import org.springframework.http.converter.HttpMessageNotReadableException;import org.springframework.http.converter.HttpMessageNotWritableException;import org.springframework.util.StreamUtils;import com.google.gson.Gson;import com.google.gson.GsonBuilder;import com.google.gson.reflect.TypeToken;public class GSONHttpMessageConverte extends AbstractHttpMessageConverter<Object> { public static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8"); public static final String DEFAULT_DATEFORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSZ"; private GsonBuilder gsonBuilder = new GsonBuilder(); @Getter @Setter private final Charset availableCharsets; @Getter @Setter private boolean writeAcceptCharset = true; @Getter @Setter private String dateFormat; @Getter @Setter private boolean excludeFieldsWithoutExposeAnnotation = false; public GSONHttpMessageConverte() { this(DEFAULT_CHARSET); } public GSONHttpMessageConverte(Charset defaultCharset) { super(new MediaType("application", "json", defaultCharset)); this.availableCharsets = DEFAULT_CHARSET; this.dateFormat = DEFAULT_DATEFORMAT; } @Override protected boolean supports(Class<?> clazz) { return true; } @Override protected Object readInternal(Class<? extends Object> clazz, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException { Charset charset = this.getContentTypeCharset(inputMessage.getHeaders().getContentType()); return StreamUtils.copyToString(inputMessage.getBody(), charset); } @Override protected void writeInternal(Object o, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException { if (this.writeAcceptCharset) { outputMessage.getHeaders().setAcceptCharset(this.getAcceptedCharsets()); } Charset charset = this.getContentTypeCharset(outputMessage.getHeaders().getContentType()); StreamUtils.copy(this.toJson(o), charset, outputMessage.getBody()); } @Override protected Long getContentLength(Object t, MediaType contentType) throws IOException { return super.getContentLength(t, contentType); } private Charset getContentTypeCharset(MediaType contentType) { if (contentType != null && contentType.getCharSet() != null) { return contentType.getCharSet(); } else { return this.availableCharsets; } } protected List<Charset> getAcceptedCharsets() { List<Charset> charsetList = new ArrayList<Charset>(1); charsetList.add(this.availableCharsets); return charsetList; } private String toJson(Object o) { try { if (this.excludeFieldsWithoutExposeAnnotation) { this.gsonBuilder = this.gsonBuilder.excludeFieldsWithoutExposeAnnotation(); } Type genericType = TypeToken.get(o.getClass()).getType(); Gson gson = gsonBuilder.setDateFormat(this.dateFormat).create(); String json = gson.toJson(o, genericType); if (json == null || "".equals(json.trim())) { json = "{}"; } return json; } catch (Exception e) { e.printStackTrace(); } return ""; }}
添加UserController的请求处理方法
直接上代码。在UserController中添加用于测试方法。为了和之前的区分访问URL我使用.json来控制。
@RequestMapping(path = "/userAdd.json", method = RequestMethod.POST, produces = "application/json; charset=UTF-8") @ResponseBody public Result userAddJson(UserMode user) { XModelAndView modeView = new XModelAndView(); modeView.addObject("user", user); System.out.println(String.format("添加用户%s完成", user.getName())); return modeView.toResult(); } @RequestMapping(path = "/userQuery.json", method = RequestMethod.GET, produces = "application/json; charset=UTF-8") @ResponseBody public Result userQueryJson(String username) { XModelAndView modeView = new XModelAndView(); UserMode user = new UserMode(); { user.setUsername(username); user.setName("小红"); user.setAge(22); user.setSex(0); } modeView.addObject("user", user); System.out.println(String.format("查询用户名为%s用户 信息成功", user.getUsername())); return modeView.toResult(); }
cn.vfire.framework.spring.mvc.controller.UserController
package cn.vfire.framework.spring.mvc.controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.ResponseBody;import org.springframework.web.bind.annotation.RestController;import cn.vfire.framework.spring.mvc.mode.UserMode;import cn.vfire.framework.spring.mvc.view.Result;import cn.vfire.framework.spring.mvc.view.XModelAndView;@RequestMapping("/api")@RestControllerpublic class UserController { @RequestMapping(path = "/userAdd.api", method = RequestMethod.POST, produces = "application/json; charset=UTF-8") @ResponseBody public String userAdd(UserMode user) { XModelAndView modeView = new XModelAndView(); modeView.addObject("user", user); System.out.println(String.format("添加用户%s完成", user.getName())); return modeView.toJson(); } @RequestMapping(path = "/userQuery.api", method = RequestMethod.GET) public String userQuery(String username) { XModelAndView modeView = new XModelAndView(); UserMode user = new UserMode(); { user.setUsername(username); user.setName("小红"); user.setAge(22); user.setSex(0); } modeView.addObject("user", user); System.out.println(String.format("查询用户名为%s用户 信息成功", user.getUsername())); return modeView.toJson(); } @RequestMapping(path = "/userSave.api", method = RequestMethod.POST) public String userSave(UserMode user) { XModelAndView modeView = new XModelAndView(); user.setAge(23); modeView.addObject("user", user); System.out.println(String.format("更新用户名为%s用户信息成功", user.getUsername())); return modeView.toJson(); } @RequestMapping(path = "/userDel.api", method = RequestMethod.GET) public String userDel(String username) { XModelAndView modeView = new XModelAndView(); System.out.println(String.format("删除用户名为%s用户信息成功", username)); return modeView.toJson(); } @RequestMapping(path = "/userAdd.json", method = RequestMethod.POST, produces = "application/json; charset=UTF-8") @ResponseBody public Result userAddJson(UserMode user) { XModelAndView modeView = new XModelAndView(); modeView.addObject("user", user); System.out.println(String.format("添加用户%s完成", user.getName())); return modeView.toResult(); } @RequestMapping(path = "/userQuery.json", method = RequestMethod.GET, produces = "application/json; charset=UTF-8") @ResponseBody public Result userQueryJson(String username) { XModelAndView modeView = new XModelAndView(); UserMode user = new UserMode(); { user.setUsername(username); user.setName("小红"); user.setAge(22); user.setSex(0); } modeView.addObject("user", user); System.out.println(String.format("查询用户名为%s用户 信息成功", user.getUsername())); return modeView.toResult(); }}
测试结果
- Spring 4.2.4.RELEASE MVC 学习笔记 - 7 - RestFull API(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 6 - RestFull API(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 1 - Maven(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 2 - Maven(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 3(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 4 - FreeMarker(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 4.1(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 5 - Log4j(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 6.1 - (咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 6.2 - (咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 6.3 - 中文乱码(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 8 - fasterxml jackson(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 8.1 - text/gson(咋个办呢 zgbn)
- Spring 4.2.4.RELEASE MVC 学习笔记 - 9 - MedaiType(咋个办呢 zgbn)
- spring mvc restfull webservice
- Spring MVC 学习笔记
- spring MVC学习笔记
- Spring MVC学习笔记
- iOS 使用断言NSAssert()调试程序错误
- POJ 1036 Gangsters(DP)
- Docker学习总结之Run命令介绍
- 如何判断自己是否具有成为一名优秀程序员的潜质及如何成为一名优秀的程序员
- static和全局变量和局部变量的问题
- Spring 4.2.4.RELEASE MVC 学习笔记 - 7 - RestFull API(咋个办呢 zgbn)
- 【手把手教你全文检索】Apache Lucene初探
- 【转载】response对象 encodeURL与encodeRedirectURL
- php-fpm进程数优化方法
- nginx 反向代理和正向代理区别
- Java 线程面试问题
- 【PHP学习】move_uploaded_file的使用
- 包管理工具之npm 和bower
- Android kernel log获取