`
jinnianshilongnian
  • 浏览: 21434996 次
  • 性别: Icon_minigender_1
博客专栏
5c8dac6a-21dc-3466-8abb-057664ab39c7
跟我学spring3
浏览量:2405130
D659df3e-4ad7-3b12-8b9a-1e94abd75ac3
Spring杂谈
浏览量:2997788
43989fe4-8b6b-3109-aaec-379d27dd4090
跟开涛学SpringMVC...
浏览量:5631528
1df97887-a9e1-3328-b6da-091f51f886a1
Servlet3.1规范翻...
浏览量:257583
4f347843-a078-36c1-977f-797c7fc123fc
springmvc杂谈
浏览量:1593212
22722232-95c1-34f2-b8e1-d059493d3d98
hibernate杂谈
浏览量:248982
45b32b6f-7468-3077-be40-00a5853c9a48
跟我学Shiro
浏览量:5847622
Group-logo
跟我学Nginx+Lua开...
浏览量:698184
5041f67a-12b2-30ba-814d-b55f466529d5
亿级流量网站架构核心技术
浏览量:780495
社区版块
存档分类
最新评论

springmvc 3.2 @MatrixVariable bug 2

 
阅读更多

之前遇到过一个bug,《spring3.2 带matrix变量的URL匹配问题》(spring3.2.3已经修复该bug),今天看到问答又有人遇到一个,在此记录下,bug可真不少,测试用例看了下,写的并不是很全面。

 

问题:

http://www.iteye.com/problems/95247

 

@RequestMapping(value = "/owners/{ownerId}/pets/{petId}", method = RequestMethod.GET)  
    public void findPet(@MatrixVariable Map<String, String> matrixVars, @MatrixVariable(pathVar = "petId") Map<String, String> petMatrixVars) {  
        System.out.println(matrixVars);  
        System.out.println(petMatrixVars);  
    }  

 上面代码是spring文档中的例子,浏览器中输入:http://localhost:8080/owners/44/pets/55;q=22,33;s=23时,控制台输出:

 

{q=[22, 33], s=[23]}  
{q=[22, 33], s=[23]}  

但是当输入http://localhost:8080/owners/42;q=11;r=12/pets/55;q=22,33;s=23则findPet方法没有执行,难道spring文档有错误?还是少了什么配置?

 

目前不支持

 

分析:

1、先看下springmvc官网的单元测试用例

@Test
	public void getLookupPathWithSemicolonContent() {
		helper.setRemoveSemicolonContent(false);

		request.setContextPath("/petclinic");
		request.setServletPath("/main");
		request.setRequestURI("/petclinic;a=b/main;b=c/welcome.html;c=d");

		assertEquals("/welcome.html;c=d", helper.getLookupPathForRequest(request));
	}

	@Test
	public void getLookupPathWithSemicolonContentAndNullPathInfo() {
		helper.setRemoveSemicolonContent(false);

		request.setContextPath("/petclinic");
		request.setServletPath("/welcome.html");
		request.setRequestURI("/petclinic;a=b/welcome.html;c=d");

		assertEquals("/welcome.html;c=d", helper.getLookupPathForRequest(request));
	}

 https://github.com/SpringSource/spring-framework/blob/master/spring-web/src/test/java/org/springframework/web/util/UrlPathHelperTests.java

 

此处可以看到,如果请求的uri是/owners/42;q=11;r=12/pets/55;q=22,33;s=23,那么request.getServletPath()实际是/owners/42,而不是期待的/owners/42/pets/55;可以参考

https://java.net/jira/browse/SERVLET_SPEC-67?page=com.atlassian.streams.streams-jira-plugin%3Aactivity-stream-issue-tab

 

接下来看下springmvc如何进行url匹配的,假设此处使用的是RequestMappingHandlerMapping(http://jinnianshilongnian.iteye.com/blog/1682510):

1、

protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {
		String lookupPath = getUrlPathHelper().getLookupPathForRequest(request);
		if (logger.isDebugEnabled()) {
			logger.debug("Looking up handler method for path " + lookupPath);
		}

		HandlerMethod handlerMethod = lookupHandlerMethod(lookupPath, request);

 此处委托给UrlPathHelper去查找path

2、

public String getLookupPathForRequest(HttpServletRequest request) {
		// Always use full path within current servlet context?
		if (this.alwaysUseFullPath) {
			return getPathWithinApplication(request);
		}
		// Else, use path within current servlet mapping if applicable
		String rest = getPathWithinServletMapping(request);
		if (!"".equals(rest)) {
			return rest;
		}
		else {
			return getPathWithinApplication(request);
		}
	}

 此处因为默认不开启alwaysUseFullPath,因此会执行getPathWithinServletMapping()

3、

public String getPathWithinServletMapping(HttpServletRequest request) {
		String pathWithinApp = getPathWithinApplication(request);
		String servletPath = getServletPath(request);
		String path = getRemainingPath(pathWithinApp, servletPath, false);
		if (path != null) {
			// Normal case: URI contains servlet path.
			return path;
		}
		else {
			// Special case: URI is different from servlet path.
			// Can happen e.g. with index page: URI="/", servletPath="/index.html"
			// Use path info if available, as it indicates an index page within
			// a servlet mapping. Otherwise, use the full servlet path.
			String pathInfo = request.getPathInfo();
			return (pathInfo != null ? pathInfo : servletPath);
		}
	}

此处调用String servletPath = getServletPath(request); 出问题了,如果我们的url是 http://localhost:8080/owners/44;a=123/pets/55;q=22,33;s=23,那么获取的将是/owners/44,而不是/owners/42/pets/55,此处有说明:

https://java.net/jira/browse/SERVLET_SPEC-67?page=com.atlassian.streams.streams-jira-plugin%3Aactivity-stream-issue-tab

 

接着执行getRemainingPath则获取到剩余部分 /pets/55;q=22,33;s=23,即和之前的单元测试一样。 所以得到的path不可能匹配模式/owners/{ownerId}/pets/{petId},但是匹配/pets/{petId}(但没有意义了),所以失败了。

 

@RequestMapping(value = "/pets/{petId}", method = RequestMethod.GET)  
    public void findPet(@MatrixVariable Map<String, String> matrixVars, @MatrixVariable(pathVar = "petId") Map<String, String> petMatrixVars) {  
        System.out.println(matrixVars);  
        System.out.println(petMatrixVars);  
    }  

 

已提交给springsource。

https://jira.springsource.org/browse/SPR-10577

9
9
分享到:
评论
14 楼 wangxiang243 2013-06-09  
jinnianshilongnian 写道
http://jinnianshilongnian.iteye.com/blog/1831408

这个修复了

谢谢你
13 楼 jinnianshilongnian 2013-06-09  
http://jinnianshilongnian.iteye.com/blog/1831408

这个修复了
12 楼 wangxiang243 2013-06-08  
jinnianshilongnian 写道
wangxiang243 写道
jinnianshilongnian 写道
wangxiang243 写道
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?

你怎么用的?


@RequestMapping(value = "/owners/{ownerId}/pets/{petId}", method = RequestMethod.GET)
	public void findPet(@MatrixVariable Map<String, String> matrixVars,
			@MatrixVariable(pathVar = "petId") Map<String, String> petMatrixVars) {
		logger.error("matrixVars:" + matrixVars);
		logger.error("petMatrixVars" + petMatrixVars);
	}


http://localhost:8080/owners/42;q=11;r=12/pets/55;q=22,33;s=23

进不到controller

这个是bug 



啊!? 不是修复了吗?如果修复的不是这个bug,你们说的修复的什么bug?
11 楼 jinnianshilongnian 2013-06-08  
wangxiang243 写道
jinnianshilongnian 写道
wangxiang243 写道
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?

你怎么用的?


@RequestMapping(value = "/owners/{ownerId}/pets/{petId}", method = RequestMethod.GET)
	public void findPet(@MatrixVariable Map<String, String> matrixVars,
			@MatrixVariable(pathVar = "petId") Map<String, String> petMatrixVars) {
		logger.error("matrixVars:" + matrixVars);
		logger.error("petMatrixVars" + petMatrixVars);
	}


http://localhost:8080/owners/42;q=11;r=12/pets/55;q=22,33;s=23

进不到controller

这个是bug 
10 楼 wangxiang243 2013-06-08  
jinnianshilongnian 写道
wangxiang243 写道
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?

你怎么用的?


@RequestMapping(value = "/owners/{ownerId}/pets/{petId}", method = RequestMethod.GET)
	public void findPet(@MatrixVariable Map<String, String> matrixVars,
			@MatrixVariable(pathVar = "petId") Map<String, String> petMatrixVars) {
		logger.error("matrixVars:" + matrixVars);
		logger.error("petMatrixVars" + petMatrixVars);
	}


http://localhost:8080/owners/42;q=11;r=12/pets/55;q=22,33;s=23

进不到controller
9 楼 jinnianshilongnian 2013-06-08  
wangxiang243 写道
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?

你怎么用的?
8 楼 wangxiang243 2013-06-08  
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?
7 楼 wangxiang243 2013-06-08  
我用的是spring3.2.3,也开启了enableMatrixVariables,怎么还是不行啊?
6 楼 jinnianshilongnian 2013-05-21  
spring 3.2.3 默认是禁用matrix variable的

可以通过<mvc:annotation-driven enableMatrixVariables="true"> 开启
5 楼 jinnianshilongnian 2013-05-21  
之前遇到过一个bug,《spring3.2 带matrix变量的URL匹配问题》(spring3.2.3已经修复该bug)

这个已经修复;

不过当前这个还是不行
4 楼 jinnianshilongnian 2013-05-21  
weishuwei 写道
对于/owners/44;m=1/pets/55;q=22,33;s=23,getServletPath不应该是只取路径,不取参数的么??没用过martin变量的场景,个人觉得后面一部分应该算参数

getServletPath取到的是/owners/44 即忽略“;”后面的,如使用jsessionid的场景;因此就出现这个问题了,如果其使用getRequestURI 就没有问题,不过如下地址也提供了解决方案
https://java.net/jira/browse/SERVLET_SPEC-67?page=com.atlassian.streams.streams-jira-plugin%3Aactivity-stream-issue-tab
3 楼 weishuwei 2013-05-21  
对于/owners/44;m=1/pets/55;q=22,33;s=23,getServletPath不应该是只取路径,不取参数的么??没用过martin变量的场景,个人觉得后面一部分应该算参数
2 楼 jinnianshilongnian 2013-05-21  
mlw2000 写道
不担心,spring改bug的速度还是很快的

之前提交一个 好久都没反应,连个回复都没有,不过他们的spring data项目 很活跃
https://jira.springsource.org/browse/SPR-10549
1 楼 mlw2000 2013-05-21  
不担心,spring改bug的速度还是很快的

相关推荐

Global site tag (gtag.js) - Google Analytics