目录
  • url @pathvariable 变量匹配原理
    • url 中带有变量的匹配原理
    • demo
    • 调试如下
    • 总结
    • 备注
  • @pathvariable
    • @pathvariable 映射 url 绑定的占位符
    • rest

url @pathvariable 变量匹配原理

url 中带有变量的匹配原理

在设置url的路径中我们可能使用变量来提高路径的灵活性,如

@requestmapping(value="/{str}/qian",method=requestmethod.get) @responsebody public string qianstr(@pathvariable string str){ return "qianstr:"+str; }

然后在输入 http://localhost:8080/zhende/qian 路径的时候就可以匹配上 qianstr 方法,但是却一直没有细想过具体是怎么实现这个匹配的。

开始预测是通过正则表达式来匹配路径就行,但问题是我输入的是 /zhende/qian 怎么匹配到 /{str}/qian 呢,还是想不通,所以通过写个demo 调试来看怎么匹配的,过程如下:

demo

使用springboot 来写个简单的web

package com.example.demo;
import org.springframework.stereotype.controller;
import org.springframework.web.bind.annotation.pathvariable;
import org.springframework.web.bind.annotation.requestmapping;
import org.springframework.web.bind.annotation.requestmethod;
import org.springframework.web.bind.annotation.responsebody;
@controller()
@requestmapping("/get")
public class testcontroller {
    @requestmapping(value="/li", method = requestmethod.get)
    @responsebody
    public string test(){
        return "test";
    }
    @requestmapping(value="/{str}",method=requestmethod.get)
    @responsebody
    public string li(@pathvariable string str){
        return "li:"+str;
    }
    @requestmapping(value="/qian/{str}",method=requestmethod.get)
    @responsebody
    public string qianstr(@pathvariable string str){
        return "qianstr:"+str;
    }
    @requestmapping(value="/{str}/qian",method=requestmethod.get)
    @responsebody
    public string strqian(@pathvariable string str){
        return "strqian:"+str;
    }
}

上边写啦好几个路径以便我们更好的比对匹配过程。

调试如下

启动服务后输入如下路径:

可以看到 左边的调用栈中 有 dispatcherservlet 分派调用过来的,所以查看dispatcherservlet 关联代码 由mappedhandler处理的所以在dispatcherservlet 中加入断点再次调试。

进入方法中

发现handlermappings 有5中处理类型,进入 mapping.gethandler(request) 方法查看具体的执行过程

simpleurlhandlermapping 中获取的handler 为空,继续尝试第二个handlermapping 类型,

这个方法会将 request中的请求路径和服务中存在的mappings 做匹配,也就是正式的路径匹配过程。继续进入方法,

由patternsrequestcondition 来处理 服务中的pattern和请求lookuppath的匹配,路径相同就直接返回该pattern,不相同则通过 antpathmatcher 来继续处理路径匹配,

将服务中的pattern 分词,然后和请求的path 做匹配,发现/test/li 和/get/zhende/qian 匹配失败,然后继续下一个pattern 的匹配,在面对分词后的{str}的匹配时,会将{str} 转成 (.*)进行正则匹配,即完成url路径中变量的匹配

找到路径后获取相应的handlermethod,然后执行对应的目标路径方法

总结

url @pathvariable 变量的匹配是由patternsrequestcondition 来完成的,具体的是由 antpathmatcher 来处理,项目启动后会将提供的url 放入容器handlermappings 中,当请求过来时会将请求path 和handlermappings 中的pattern 逐一匹配,由匹配时会将路径先分词,有{str}变量的转成 (.*) 再匹配,匹配完成后调用相应的handlermethod执行方法,进入目标方法。

备注

遗留问题,当项目中有多个路径可以匹配会怎么执行?为什么一开始的dispatcherservlet 中会有5中类型的handlermappings?

@pathvariable

@pathvariable 映射 url 绑定的占位符

带占位符的 url 是 spring3.0 新增的功能,该功能在springmvc 向 rest 目标挺进发展过程中具有里程碑的意义

通过 @pathvariable 可以将 url 中占位符参数绑定到控制器处理方法的入参中:url 中的 {xxx} 占位符可以通过@pathvariable(“xxx“) 绑定到操作方法的入参中。

实例:

springmvctest.java

//@pathvariable可以用来映射url中的占位符到目标方法的参数中
@requestmapping("/testpathvariable/{id}")
    public string testpathvariable(@pathvariable("id") integer id)
    {
        system.out.println("testpathvariable:"+id);
        return success;
    }

index.jsp

<a href="springmvc/testpathvariable/1" rel="external nofollow" >testpathvariable</a>

rest

  • rest:即 representational state transfer。(资源)表现层状态转化。是目前最流行的一种互联网软件架构。它结构清晰、符合标准、易于理解、扩展方便,所以正得到越来越多网站的采用
  • 资源(resources):网络上的一个实体,或者说是网络上的一个具体信息。它可以是一段文本、一张图片、一首歌曲、一种服务,总之就是一个具体的存在。可以用一个uri(统一资源定位符)指向它,每种资源对应一个特定的 uri 。要获取这个资源,访问它的uri就可以,因此 uri 即为每一个资源的独一无二的识别符。
  • 表现层(representation):把资源具体呈现出来的形式,叫做它的表现层(representation)。比如,文本可以用 txt 格式表现,也可以用 html 格式、xml 格式、json 格式表现,甚至可以采用二进制格式。
  • 状态转化(state transfer):每发出一个请求,就代表了客户端和服务器的一次交互过程。http协议,是一个无状态协议,即所有的状态都保存在服务器端。因此,如果客户端想要操作服务器,必须通过某种手段,让服务器端发生“状态转化”(state transfer)。而这种转化是建立在表现层之上的,所以就是 “表现层状态转化”。具体说,就是 http 协议里面,四个表示操作方式的动词:get、post、put、delete。它们分别对应四种基本操作:get 用来获取资源,post 用来新建资源,put 用来更新资源,delete 用来删除资源。

示例:

  • - /order/1 http get :得到 id = 1 的 order
  • - /order/1 http delete:删除 id = 1的 order
  • - /order/1 http put:更新id = 1的 order
  • - /order http post:新增 order

hiddenhttpmethodfilter:浏览器 form 表单只支持 get与 post 请求,而delete、put 等 method 并不支持,spring3.0 添加了一个过滤器,可以将这些请求转换为标准的 http 方法,使得支持 get、post、put 与delete 请求。

在web.xml中配置hiddenhttpmethodfilter

web.xml

 <!--
     配置org.springframework.web.filter.hiddenhttpmethodfilter:可以把post请求转换成delete或者post请求
      -->
   <filter>
      <filter-name>hiddenhttpmethodfilter</filter-name>
      <filter-class>org.springframework.web.filter.hiddenhttpmethodfilter</filter-class>
   </filter>   
   <filter-mapping>
      <filter-name>hiddenhttpmethodfilter</filter-name>
      <url-pattern>/*</url-pattern>
   </filter-mapping>

在springmvctest中测试相应的方法

springmvctest.java

/*
     * 如何发送put和delete请求
     * 1.需要配置hiddenhttpmethodfilter
     * 2.需要发送post请求
     * 3.需要发送post请求时携带一个name="_method"的隐藏域,value值为delete或者put
     * 
     * 在springmvc框架中通过@pathvariable注解来获取id值
     * */
    //get请求
    @requestmapping(value="/testrest/{id}",method=requestmethod.get)
    public string testrest(@pathvariable integer id)
    {
        system.out.println("testrest get"+id);
        return success;
    }
    //post请求
    @requestmapping(value="/testrest",method=requestmethod.post)
    public string testrest()
    {
        system.out.println("testrest post");
        return success;
    }
    //delete请求
    @requestmapping(value="/testrest/{id}",method=requestmethod.delete)
    public string testrestdelete(@pathvariable integer id)
    {
        system.out.println("testrest delete"+id);
        return success;
    }
    //delete请求
    @requestmapping(value="/testrest/{id}",method=requestmethod.put)
    public string testrestput(@pathvariable integer id)
    {
        system.out.println("testrest put"+id);
        return success;
    }

index.jsp中的写法

index.jsp

<a href="springmvc/testrest/1" rel="external nofollow" >test rest get</a><br/>
<form action="springmvc/testrest" method="post">
<input type="submit" value="testrest post"/>
</form><br/>
<form action="springmvc/testrest/1" method="post">
<input type="hidden" name="_method" value="delete"/>
<input type="submit" value="testrest delete"/>
</form><br/>
<form action="springmvc/testrest/1" method="post">
<input type="hidden" name="_method" value="put"/>
<input type="submit" value="testrest put"/>
</form><br/>

以上为个人经验,希望能给大家一个参考,也希望大家多多支持www.887551.com。