菜鸟搭建web服务笔记2

2019-03-16  本文已影响0人  DizzyDwarf

回顾

上一篇笔者创建了一个简单的web服务,但是因为前后端路径不一致导致很难实现一个servlet处理并分发所有的url请求。不过好在struts2、SpringMVC等框架已经帮我们实现了这个功能。在这一篇中笔者将尝试用SpringMVC来改造之前的项目。

环境

改造

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.1.5.RELEASE</version>
</dependency>

路径

在开始测试之前先说一下在servlet中获取路径的相关方法,对于http://localhost:8080/project_name/resource_name

requestURI = contextPath + servletPath + pathInfo

测试

将war包部署到Tomcat然后启动测试,结果提示

org.springframework.web.servlet.DispatcherServlet.noHandlerFound
No mapping for GET /dizzydwarf-0.0.1-SNAPSHOT/login

居然说找不到/login的映射方法,仔细检查了下代码,Controller类中的@Controller@RequestMapping("/login")都没什么问题,web.xml中的url-pattern也是/login,html表单中的action是login也没问题。试着把@RequestMapping中的/login改成login,居然成功了,但是这也不科学啊。为了解答这个疑惑,笔者决定对Tomcat进行远程调试。

Tomcat远程调试

因为是第一次对Tomcat进行远程调试,这里简单介绍下配置方法

SET CATALINA_OPTS=-server -Xdebug -Xnoagent -Djava.compiler=NONE -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=8000

笔者在DispatcherServlet.getHandlerAbstractHandlerMethodMapping.getMappingsByUrl两个方法中设置了断点,然后在浏览器中输入地址发送请求以触发断点。结果出人意料的是问题居然平白无故消失了,两个地方都正常返回了那个/login映射的方法。

返回text或者json数据

这里我们不想用jsp,暂时只希望返回text或者json数据。有几种方法可以实现这个需求

方法一

给Controller方法传递HttpServletRequestHttpServletResponse,按照传统的处理思路,最后返回void。但是因为我们已经用了SpringMVC,所以还是希望以SpringMVC的方式处理

方法二

在方法前面加上一个@RequestBody,则方法的返回值将直接作为body的内容。比如说我们返回的是一个字符串,那么body里面就是一个字符串。但是如果我们返回的是一个复杂对象呢?对于复杂对象,这里希望以json格式序列化后返回。笔者试着只返回一个复杂对象,不做任何其他处理,结果提示No converter found for return value of type,看来该配置的还是不能省。

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.9.8</version>
</dependency>
public RequestMappingHandlerAdapter() {
    StringHttpMessageConverter stringHttpMessageConverter = new StringHttpMessageConverter();
    stringHttpMessageConverter.setWriteAcceptCharset(false);  // see SPR-7316

    this.messageConverters = new ArrayList<>(4);
    this.messageConverters.add(new ByteArrayHttpMessageConverter());
    this.messageConverters.add(stringHttpMessageConverter);
    try {
        this.messageConverters.add(new SourceHttpMessageConverter<>());
    }
    catch (Error err) {
        // Ignore when no TransformerFactory implementation is available
    }
    this.messageConverters.add(new AllEncompassingFormHttpMessageConverter());
}

<mvc:annotation-driven>可以使Spring调用WebMvcConfigurationSupportaddDefaultHttpMessageConverters方法

protected final void addDefaultHttpMessageConverters(List<HttpMessageConverter<?>> messageConverters) {
    StringHttpMessageConverter stringHttpMessageConverter = new StringHttpMessageConverter();
    stringHttpMessageConverter.setWriteAcceptCharset(false);  // see SPR-7316

    messageConverters.add(new ByteArrayHttpMessageConverter());
    messageConverters.add(stringHttpMessageConverter);
    messageConverters.add(new ResourceHttpMessageConverter());
    messageConverters.add(new ResourceRegionHttpMessageConverter());
    try {
        messageConverters.add(new SourceHttpMessageConverter<>());
    }
    catch (Throwable ex) {
        // Ignore when no TransformerFactory implementation is available...
    }
    messageConverters.add(new AllEncompassingFormHttpMessageConverter());

    ......

    if (jackson2Present) {
        Jackson2ObjectMapperBuilder builder = Jackson2ObjectMapperBuilder.json();
        if (this.applicationContext != null) {
            builder.applicationContext(this.applicationContext);
        }
        messageConverters.add(new MappingJackson2HttpMessageConverter(builder.build()));
    }

    ......
}

在这个方法中除了注册之前默认的messageConverter,还会加入支持json转换的messageConverter

禁用后缀模式

笔者增加了一个register.html用来注册新用户,这个页面中有一个action等于/register的表单,同时在Controller中加了一个处理方法,这个方法的@RequestMapping值是/register。结果当访问/register.html时,却发现请求直接被发送到了这个处理方法,而不是显示一个静态页面。之前笔者在配置文件中是否配置过静态资源的映射的。

<mvc:resources location="/" mapping="*.html"></mvc:resources>

一查资料,发现在SpringMVC中默认支持后缀模式,也就是说@RequestMapping("/register")会被当成/register.*来匹配。解决方法是在配置文件中加入以下内容

<mvc:annotation-driven>
    <mvc:path-matching suffix-pattern="false" />
</mvc:annotation-driven>

总结

上一篇下一篇

猜你喜欢

热点阅读