使用jvm sandbox對(duì)三層嵌套類型的改造示例
問題背景
先簡單介紹下基于jvm-sandbox
的imock工具,是Java方法級(jí)別的mock,操作就是監(jiān)聽指定方法,返回指定的mock內(nèi)容。
jvm-sandbox
利用字節(jié)碼操作和自定義類加載器的技術(shù),將原始方法替換為模擬代碼,從而在應(yīng)用程序中實(shí)現(xiàn)方法級(jí)別的模擬。這種方法非常強(qiáng)大,但也需要對(duì)字節(jié)碼操作、類加載機(jī)制和 JVM 內(nèi)部原理有一定的理解。
公司要搭建一個(gè)方法級(jí)別的后端mock平臺(tái),因此我在imock的基礎(chǔ)上進(jìn)行二次開發(fā)進(jìn)行使用。
問題描述
在mock某個(gè)三方接口的方法時(shí)遇到報(bào)錯(cuò):
ava.lang.ClassCastException: com.alibaba.fastjson.JSONObject cannot be cast to com.travelsky.angeldoe.output.PassengerFlightInfo
看樣子是本來應(yīng)該是JSONObject 無法轉(zhuǎn)化成PassengerFlightInfo類型,通過日志排查問題,定位到報(bào)錯(cuò)代碼。
PassengerFlightInfo?passengerFlightInfo?=?JSON.parseObject(out ?.getPassengerFlightInfoList().get(0).toString(),?PassengerFlightInfo.class);
線上服務(wù)沒有報(bào)錯(cuò),測試mock環(huán)境報(bào)錯(cuò),那么顯然是數(shù)據(jù)的問題,通過Arthas追蹤方法返回的bean對(duì)比發(fā)現(xiàn),差異就是線上的PassengerFlightInfo是一個(gè)bean,測試的PassengerFlightInfo是一個(gè)object。差異由此出現(xiàn)。
那么問題的關(guān)鍵就在于,如何通過mock工具把object提前轉(zhuǎn)成bean。
解決方案
改造mock agent工具思路:通過我們的mock-module.jar實(shí)現(xiàn)。
- 根據(jù)PsrInfoOutputBean初步解析returnObject,獲取list中的object
- 將object解析成PassengerFlightInfo,再通過反射技術(shù)將bean反射回PsrInfoOutputBean
代碼實(shí)現(xiàn)
//針對(duì)cki特殊類型PsrInfoOutputBean case?3: ????//獲取advice返回類型的類加載器 ????ClassLoader?behaviorClassLoader?=?advice.getBehavior().getReturnType().getClassLoader(); ????//加載最外層PsrInfoOutputBean ????Class<?>?targetClass?=?behaviorClassLoader.loadClass(ro.getClassNames()[0]); ????LogUtil.info2("targetClass=",?targetClass.toString()); ????//根據(jù)目標(biāo)類解析returnData ????Object?res1?=?JSON.parseObject(ro.getReturnData(),?targetClass); ????//賦值保存做對(duì)比 ????Object?res0?=?res1; ????LogUtil.info2("res1-before=",?res1.toString()); ????//?通過反射獲取passengerFlightInfoList ????List<Object>?passengerFlightInfoList?=?(List<Object>)?targetClass.getMethod("getPassengerFlightInfoList").invoke(res1); ????LogUtil.info2("passengerFlightInfoList=",?passengerFlightInfoList.toString()); ????if?(!passengerFlightInfoList.isEmpty())?{ ????????//?獲取?passengerFlightInfoList?列表中的第一個(gè)元素 ????????Object?firstPassengerFlightInfoList?=?passengerFlightInfoList.get(0); ????????LogUtil.info2("firstPassengerFlightInfoList=",?firstPassengerFlightInfoList.toString()); ????????//?將?firstFlightInfo?轉(zhuǎn)換成?JSON?字符串 ????????String?firstFlightInfoJson?=?JSON.toJSONString(firstPassengerFlightInfoList); ????????//?獲取第三層額外目標(biāo)?Bean?類的類名,使用同一類加載器 ????????Class<?>?targetBeanClass?=?behaviorClassLoader.loadClass(ro.getClassNames()[2]); ????????LogUtil.info2("targetBeanClass=",?targetBeanClass.toString()); ????????//根據(jù)類解析成bean ????????Object?targetBean?=?JSON.parseObject(firstFlightInfoJson,?targetBeanClass); ????????LogUtil.info2("targetBean=",?targetBean.toString()); ????????//?創(chuàng)建一個(gè)新的passengerFlightInfoListNew?將?targetBean?添加到?passengerFlightInfoList?中 ????????List<Object>?passengerFlightInfoListNew?=?new?ArrayList<>(); ????????passengerFlightInfoListNew.add(targetBean); ????????//?設(shè)置?passengerFlightInfoList?屬性回?res1 ????????try?{ ????????????//?執(zhí)行反射方法,把passengerFlightInfoListNew反射回res ????????????Method?method?=?targetClass.getMethod("setPassengerFlightInfoList",?List.class); ????????????method.invoke(res1,?passengerFlightInfoListNew); ????????}?catch?(Exception?e)?{ ????????????//?捕獲異常并打印日志 ????????????LogUtil.info2("Error?occurred?while?invoking?method:=",?e.getMessage()+"|"+e); ????????} ????} ????LogUtil.info2("前后的兩個(gè)類equals嗎?=",?String.valueOf(res1.equals(res0))); ????LogUtil.info2("res1-after=",?res1.toString()); ????ProcessController.returnImmediately(res1); ????break;
遇到的坑
外部獲取的類名不能直接通過Class.forName加載,如下代碼所示:
?//?使用目標(biāo)?Bean?類名解析?JSON?字符串成目標(biāo)?Bean ????????Class<?>?targetBeanClass?=?Class.forName(targetBeanClassName);
實(shí)際會(huì)報(bào)錯(cuò):"message": "com.taobao.rigel.rap.model.PsrInfoOutputBean cannot be cast to com.taobao.rigel.rap.model.PsrInfoOutputBean", 原因是這兩個(gè)bean雖然名字一樣,但是類加載器不同,就導(dǎo)致bean的實(shí)際是不一樣的。類是否相同可以用equals進(jìn)行判斷。
因此正確的做法是,先獲取advice返回類型的類加載器,然后加載我們所需要的類,這樣業(yè)務(wù)的代碼就會(huì)認(rèn)得我們的bean了。
???//獲取advice返回類型的類加載器 ????ClassLoader?behaviorClassLoader?=?advice.getBehavior().getReturnType().getClassLoader(); ????//加載最外層PsrInfoOutputBean ????Class<?>?targetClass?=?behaviorClassLoader.loadClass(ro.getClassNames()[0]);
題外話:
為啥出現(xiàn)了這個(gè)錯(cuò)誤?
出現(xiàn)這個(gè)報(bào)錯(cuò)和開發(fā)的強(qiáng)轉(zhuǎn)類型也有關(guān)系,本地做了個(gè)小測試,同樣的數(shù)據(jù)。(但咱也沒發(fā)改開發(fā)的代碼,只能提提建議。 = =)
1、當(dāng)前異常轉(zhuǎn)化:按照開發(fā)業(yè)務(wù)代碼中的list強(qiáng)轉(zhuǎn)對(duì)象
List<Object> list = JSON.*parseArray*(jsonString); PassengerFlightInfo passengerFlightInfo = (PassengerFlightInfo) list.get(0);
這是使用強(qiáng)制類型轉(zhuǎn)換的方式,直接將 list
中的第一個(gè)元素強(qiáng)制轉(zhuǎn)換為 PassengerFlightInfo
對(duì)象。
這種方式在編譯時(shí)不會(huì)報(bào)錯(cuò),但如果 list
中的第一個(gè)元素不是 PassengerFlightInfo
對(duì)象,則會(huì)在運(yùn)行時(shí)拋出 ClassCastException
異常。
2、正常轉(zhuǎn)化:優(yōu)化過后用toJavaObject方法
PassengerFlightInfo passengerFlightInfo = ((JSONObject) list.get(0)).toJavaObject(PassengerFlightInfo.class);
這是使用 FastJSON 提供的 toJavaObject
方法,將 JSONObject
類型轉(zhuǎn)換為 PassengerFlightInfo
對(duì)象。
這種方式在運(yùn)行時(shí)會(huì)檢查轉(zhuǎn)換是否可行,如果 JSONObject
不包含 PassengerFlightInfo
的屬性或結(jié)構(gòu)不匹配,會(huì)拋出異常。這種方式更安全,因?yàn)樗峁┝烁嗟霓D(zhuǎn)換檢查。
推薦使用第二種方式,因?yàn)樗咏押桶踩?,能夠更好地處理可能出現(xiàn)的異常情況,并提供更好的錯(cuò)誤信息。
以上就是使用jvm sandbox對(duì)三層嵌套類型的改造示例的詳細(xì)內(nèi)容,更多關(guān)于jvm sandbox改造三層嵌套類型的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Mybatis-Plus中update()和updateById()將字段更新為null
本文主要介紹了Mybatis-Plus中update()和updateById()將字段更新為null,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08Java中的StringTokenizer實(shí)現(xiàn)字符串切割詳解
這篇文章主要介紹了Java中的StringTokenizer實(shí)現(xiàn)字符串切割詳解,java.util工具包提供了字符串切割的工具類StringTokenizer,Spring等常見框架的字符串工具類(如Spring的StringUtils),需要的朋友可以參考下2024-01-01Java中json使用方法_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
JSON(JavaScript Object Notation) 是一種輕量級(jí)的數(shù)據(jù)交換格式, json是個(gè)非常重要的數(shù)據(jù)結(jié)構(gòu),在web開發(fā)中應(yīng)用十分廣泛。下面通過本文給大家講解Java中json使用方法,感興趣的朋友一起看看吧2017-07-07Spring?Boot中@Validated注解不生效問題匯總大全
這篇文章主要給大家介紹了關(guān)于Spring?Boot中@Validated注解不生效問題匯總的相關(guān)資料,@Validated注解是Spring框架中的一個(gè)注解,用于在方法參數(shù)上添加參數(shù)校驗(yàn)規(guī)則,需要的朋友可以參考下2023-07-07Spring Boot啟動(dòng)流程斷點(diǎn)過程解析
這篇文章主要介紹了Spring Boot啟動(dòng)流程斷點(diǎn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11Java ArrayList與LinkedList及HashMap容器的用法區(qū)別
這篇文章主要介紹了Java ArrayList與LinkedList及HashMap容器的用法區(qū)別,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-07-07Spring Boot集成Spring Cloud Security進(jìn)行安全增強(qiáng)的方法
Spring Cloud Security是Spring Security的擴(kuò)展,它提供了對(duì)Spring Cloud體系中的服務(wù)認(rèn)證和授權(quán)的支持,包括OAuth2、JWT等,這篇文章主要介紹了Spring Boot集成Spring Cloud Security進(jìn)行安全增強(qiáng),需要的朋友可以參考下2024-11-11