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();    }}

测试结果

URL 测试页面 测试结果 http://127.0.0.1:8000/api/userAdd.json 这里写图片描述 这里写图片描述 http://127.0.0.1:8000/api/userQuery.json?username=xiaohong 这里写图片描述 这里写图片描述
0 0
原创粉丝点击