spring mvc中的@PathVariable動(dòng)態(tài)參數(shù)詳解
spring mvc @PathVariable動(dòng)態(tài)參數(shù)
spring mvc中的@PathVariable是用來獲得請(qǐng)求url中的動(dòng)態(tài)參數(shù)的,十分方便
@Controller public class TestController { @RequestMapping(value="/user/{userId}/roles/{roleId}",method = RequestMethod.GET) public String getLogin(@PathVariable("userId") String userId, @PathVariable("roleId") String roleId){ System.out.println("User Id : " + userId); System.out.println("Role Id : " + roleId); return "hello"; } @RequestMapping(value="/product/{productId}",method = RequestMethod.GET) public String getProduct(@PathVariable("productId") String productId){ System.out.println("Product Id : " + productId); return "hello"; } @RequestMapping(value="/javabeat/{regexp1:[a-z-]+}", method = RequestMethod.GET) public String getRegExp(@PathVariable("regexp1") String regexp1){ System.out.println("URI Part 1 : " + regexp1); return "hello"; } }
spring mvc是如何做到根據(jù)參數(shù)名動(dòng)態(tài)綁定參數(shù)的?
使用過SpringMVC的同學(xué)都知道,當(dāng)我們需要在Controller層接收客戶端的請(qǐng)求參數(shù)時(shí),只需要在形參上加@RequestParam注解,SpringMVC就會(huì)自動(dòng)幫我們做參數(shù)綁定,如下示例:
@GetMapping("test1") public void test1(@RequestParam("name") String name, @RequestParam("age") Integer age) { }
客戶端請(qǐng)求示例:
curl http://127.0.0.1:8080/test1?name=root&age=18
每個(gè)參數(shù)都加注解寫起來非常的麻煩,因此SpringMVC還可以根據(jù)參數(shù)名自動(dòng)匹配,只要方法的參數(shù)名和客戶端請(qǐng)求的參數(shù)名相同即可綁定,代碼可以簡(jiǎn)化為:
@GetMapping("test2") public void test2(String name, Integer age) throws Exception { }
SpringMVC是如何做到的呢???
反射獲取參數(shù)名
熟悉SpringMVC的同學(xué)都知道,SpringMVC通過一個(gè)DispatcherServlet來分發(fā)客戶端的請(qǐng)求,根據(jù)請(qǐng)求的URI映射對(duì)應(yīng)的處理器Handler,將請(qǐng)求交給對(duì)應(yīng)的Handler處理,說白了就是通過反射的方式調(diào)用Controller的方法,然后將請(qǐng)求的參數(shù)解析,并和方法的形參做匹配并傳遞過去。
要想綁定參數(shù),首先要做的就是知曉Controller的方法需要的參數(shù)名是什么???
對(duì)于第一種寫法,很好理解,方法想要的參數(shù)名就是@RequestParam注解的值,只需要通過反射來獲取即可,如下代碼:
public static void main(String[] args) throws Exception { Method test1 = UserController.class.getMethod("test1", String.class, Integer.class); for (Parameter parameter : test1.getParameters()) { RequestParam requestParam = parameter.getAnnotation(RequestParam.class); System.err.println("test1-參數(shù)名:" + requestParam.value()); } } 控制臺(tái)輸出: test1-參數(shù)名:name test1-參數(shù)名:age
但是對(duì)于第二種簡(jiǎn)化的寫法,是無法通過反射來獲取參數(shù)名稱的,如下:
public static void main(String[] args) throws Exception { Method test2 = UserController.class.getMethod("test2", String.class, Integer.class); for (Parameter parameter : test2.getParameters()) { System.err.println("test2-參數(shù)名:"+parameter.getName()); } }
你們猜猜拿到的參數(shù)名是什么???
竟然是沒有任何意義的arg0、arg1!??!
這是為什么呢???
熟悉JVM的同學(xué)都知道,Java代碼要想在JVM里執(zhí)行,首先需要通過javac命令編譯成字節(jié)碼Class文件,而這個(gè)編譯的過程會(huì)直接將方法的參數(shù)名稱丟棄,變成無意義的arg0、arg1…,因此通過反射是無法獲取參數(shù)名稱的。
-parameters參數(shù)
既然反射獲取不到參數(shù)名是因?yàn)榫幾g時(shí)丟棄了,那么有沒有辦法讓javac編譯時(shí)將參數(shù)名保留下來呢???答案是有的,那就是-parameters參數(shù)。
JDK8加入了一個(gè)新功能,編譯時(shí)加上-parameters參數(shù),即可保留參數(shù)名,通過parameter.getName()就可以獲取到正常的參數(shù)名了。
示例
有如下測(cè)試類:
public class Demo { public void test(String name, Integer age) { } }
javac Demo.java #默認(rèn)的編譯方式 javap -verbose Demo
javac -parameters Demo.java #加-parameters參數(shù)編譯 javap -verbose Demo
可以看到,加了-parameters參數(shù)后,字節(jié)碼文件會(huì)使用額外的MethodParameters區(qū)域來保存方法的參數(shù)名稱。這樣反射的時(shí)候通過parameter.getName()就可以獲取到參數(shù)名了。
注意:只支持JDK8及以上版本?。?!
-g參數(shù)
由于-parameters要求JDK至少是8版本,而SpringMVC肯定是要支持低版本JDK的,那么還有沒有其他方法可以保留參數(shù)名呢???
答案依然是有的,那就是-g參數(shù)。
編譯時(shí),加上-g參數(shù)就是告訴編譯器,我們需要調(diào)試類的信息,這時(shí)編譯器在編譯時(shí),就會(huì)保留局部變量表的信息,參數(shù)也是局部變量表的一部分。
可以看到,加上-g后就可以從局部變量表中獲取參數(shù)的名稱了。
使用Maven來管理項(xiàng)目的話,編譯會(huì)默認(rèn)加-g參數(shù),不需要開發(fā)者介入。
注意:雖然-g會(huì)將局部變量表的信息保存下來,但是依然無法通過反射parameter.getName()的方式來獲取參數(shù)名,需要開發(fā)者去解析Class字節(jié)碼文件來獲取,這是和-parameters的一個(gè)重大區(qū)別!!!
ASM
ASM是一個(gè)通用的Java字節(jié)碼操作和分析框架。 它可以用于修改現(xiàn)有類或直接以二進(jìn)制形式動(dòng)態(tài)生成類。 ASM提供了一些常見的字節(jié)碼轉(zhuǎn)換和分析算法,可以從中構(gòu)建自定義復(fù)雜轉(zhuǎn)換和代碼分析工具。 ASM提供與其他Java字節(jié)碼框架類似的功能,但專注于性能。 因?yàn)樗脑O(shè)計(jì)和實(shí)現(xiàn)盡可能小而且快,所以它非常適合在動(dòng)態(tài)系統(tǒng)中使用(但當(dāng)然也可以以靜態(tài)方式使用,例如在編譯器中)。
編譯時(shí)加上-g參數(shù)可以將參數(shù)名保留下來,但是依然無法通過反射來獲取,需要解析字節(jié)碼文件自己獲取。
有沒有好用的工具包來幫我們解析字節(jié)碼文件呢???
答案依然是:有的。
Java通過ASM就可以很方便的操作字節(jié)碼文件,很多開源框架都用到了ASM,例如CGLIB。
下面寫一個(gè)例子,通過ASM來獲取方法的參數(shù)名。
1、引入依賴
<dependency> <groupId>asm</groupId> <artifactId>asm-util</artifactId> <version>3.3.1</version> </dependency>
2、代碼示例
public class Demo { public void test(String name, Integer age) { } /** * 通過ASM來訪問參數(shù)名 * @param args * @throws Exception */ public static void main(String[] args) throws Exception { Class<Demo> clazz = Demo.class; Method method = clazz.getMethod("test", String.class, Integer.class); InputStream in = clazz.getResourceAsStream("/" + clazz.getName().replace('.', '/') + ".class"); ClassReader cr = new ClassReader(in); ClassNode cn = new ClassNode(); cr.accept(cn, ClassReader.EXPAND_FRAMES); List<MethodNode> methodNodes = cn.methods; for (MethodNode methodNode : methodNodes) { if (method.getName().equals(methodNode.name)) { System.err.println("test方法參數(shù):"); List<LocalVariableNode> localVariables = methodNode.localVariables; for (LocalVariableNode localVariable : localVariables) { System.err.println(localVariable.name); } } } } }
控制臺(tái)輸出:
test方法參數(shù):
this
name
age
注意:這種方式對(duì)接口和抽象方法沒有用,因?yàn)槌橄蠓椒]有方法體,也就沒有局部變量表。這也就是為什么MyBatis在xml中無法根據(jù)接口方法的參數(shù)名去綁定參數(shù)的原因!!!
至此,我們已經(jīng)知道,Java獲取方法的參數(shù)名有兩種方式,分別是加-parameters參數(shù)反射獲取、-g參數(shù)通過ASM解析字節(jié)碼文件獲取。
那SpringMVC用的是哪種呢???
SpringMVC的處理方式
SpringMVC是如何解決參數(shù)名稱的問題的呢?是通過-parameters參數(shù)嗎???
當(dāng)然不是,首先-parameters參數(shù)是JDK8才提供的,老版本的JDK根本沒這個(gè)功能,SpringMVC是要支持JDK8之前的版本的,而且這種解決方案強(qiáng)制要求開發(fā)者編譯時(shí)手動(dòng)加參數(shù),也很不友好。
要想知道SpringMVC的解決方案,必須看源碼?。?!
Debug跟蹤源碼的過程筆者就不詳敘了,感興趣的同學(xué)可以自己去跟蹤一下。
SpringMVC將一個(gè)方法處理器封裝為一個(gè)HandlerMethod類,方法的參數(shù)則用MethodParameter表示:
MethodParameter有一個(gè)獲取參數(shù)名的方法getParameterName():
獲取參數(shù)名的的任務(wù)其實(shí)是交給ParameterNameDiscoverer去完成了,這是一個(gè)接口,主要的作用就是解析方法的參數(shù)名稱。
MethodParameter的ParameterNameDiscoverer實(shí)現(xiàn)類是PrioritizedParameterNameDiscoverer。
距離真相只剩一步之遙了,去看看LocalVariableTableParameterNameDiscoverer實(shí)現(xiàn)吧。
只要看inspectClass()方法就知道真相了。
可以看到,LocalVariableTableParameterNameDiscoverer底層就是用的ASM的技術(shù)來獲取方法的參數(shù)名的。只是Spring并沒有直接依賴ASM,而是將他們封裝到了自己的org.springframework.asm包下。
總結(jié)
SpringMVC獲取Controller方法的參數(shù)名有三種方式,如下:
方案 | 限制 | 優(yōu)缺點(diǎn) |
---|---|---|
參數(shù)加注解 | 不受限 | 編寫麻煩 |
-parameters | JDK8及以上才支持 | 直接通過parameter.getName()獲取,方便 |
-g | 不受限,編譯加-g參數(shù)即可 | 解析比較麻煩,依賴于ASM |
- 如果加了@RequestParam則優(yōu)先使用注解解析。
- 如果沒有注解,則采用StandardReflectionParameterNameDiscoverer解析,通過Parameter.getName()反射獲取,前提是JDK版本為8以上,且開啟了-parameters編譯參數(shù)。
- 如果前面2種都無法獲取,則采用LocalVariableTableParameterNameDiscoverer通過ASM技術(shù)來解析。
注意:如果編譯不加-g參數(shù),即使是用ASM也無法解析,巧婦難為無米之炊?。。?/p>
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
spring cloud-給Eureka Server加上安全的用戶認(rèn)證詳解
這篇文章主要介紹了spring cloud-給Eureka Server加上安全的用戶認(rèn)證詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-01-01Java通過MyBatis框架對(duì)MySQL數(shù)據(jù)進(jìn)行增刪查改的基本方法
MyBatis框架由Java的JDBC API進(jìn)一步封裝而來,在操作數(shù)據(jù)庫方面效果拔群,接下來我們就一起來看看Java通過MyBatis框架對(duì)MySQL數(shù)據(jù)進(jìn)行增刪查改的基本方法:2016-06-06Springboot2.1.6集成activiti7出現(xiàn)登錄驗(yàn)證的實(shí)現(xiàn)
這篇文章主要介紹了Springboot2.1.6集成activiti7出現(xiàn)登錄驗(yàn)證的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12Java Web開發(fā)中過濾器和監(jiān)聽器使用詳解
這篇文章主要為大家詳細(xì)介紹了Java中的過濾器Filter和監(jiān)聽器Listener的使用以及二者的區(qū)別,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-10-10JavaMail實(shí)現(xiàn)簡(jiǎn)單郵件發(fā)送
這篇文章主要為大家詳細(xì)介紹了JavaMail實(shí)現(xiàn)簡(jiǎn)單郵件發(fā)送,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08Spring中bean的生命周期之getSingleton方法
今天給大家?guī)淼氖顷P(guān)于Spring的相關(guān)知識(shí),文章圍繞著Spring中bean的生命周期之getSingleton方法展開,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06