如今網上大多數是使用TypeReference
方式來解析JSON數據,這裏我提供另一種方式來解析,使用類文件進行解析,兩種方式我都會給出實際代碼java
@GET
@Path("getUserCourse")
@Produces(MediaType.APPLICATION_JSON)
public Result<List<UserCourseDto>> getUserCourse(){
return externalBiz.getUserCourse();
}
public Result<List<UserCourseDto>> getUserCourse(){
String result = restTemplate.getForObject(MOCK_JSON_URL, String.class);
return JSONObject.parseObject(result, new TypeReference<Result<List<UserCourseDto>>>() {});
}
這裏使用的基於jsr-339標準的resteasy開發demojson
瀏覽器訪問:http://localhost:8080/v1/province/getUserCourse
能夠獲得json返回的結果數組
{
status: 0,
message: "1",
data: [
{
openType: "兌換 ",
userID: 30014,
classID: 10376,
className: "趣味職場俚語課程【11月班】",
chargeFee: 106,
classStudyTime: null,
openRMB: 0,
rechargeFee: 0,
awardFee: 0,
openFee: 0,
dateAdded: 1312175789393,
expiredDate: 1323964800000
}
]
}
介紹:瀏覽器
這裏使用了提供的TypeReference進行包裝,可以清晰明瞭進行解析多級泛型,可是有時候,咱們作一些通用的解析框架的時候,可能會用T類型,T類型拿到的是字節碼文件,或者class對象,又該怎麼處理呢?請看下面介紹。app
接口:框架
接口之類用了一個註解來處理的,沒有直接傳class對象過去,由於在實際項目中,基本都是註解,沒有誰會直接傳class對象。因此我傳的Annotation數組過去啦this
@GET
@Reader(extParamClass = {Result.class,List.class,UserCourseDto.class})
@Path("getUserCourseV2")
@Produces(MediaType.APPLICATION_JSON)
public Result<List<UserCourseDto>> getUserCourseV2(){
Annotation[] annotations = new Annotation[0];
for (Method method : this.getClass().getMethods()) {
if (method.getName().equals("getUserCourseV2")){
annotations = method.getAnnotations();
}
}
return externalBiz.getUserCourseV2(annotations);
}
處理:spa
public Result<List<UserCourseDto>> getUserCourseV2(Annotation[] annotations) {
final Reader[] readers = {null};
if(annotations != null) {
Arrays
.stream(annotations)
.filter(annotation -> annotation.annotationType().equals(Reader.class))
.findFirst().ifPresent(x -> readers[0] = (Reader) x);
}
Class[] classes = readers[0].extParamClass();
String result = restTemplate.getForObject(MOCK_JSON_URL, String.class);
//這裏不用TypeReference方式,直接用class對象來處理
ParameterizedTypeImpl beforeType = null;
if (classes.length!=0){
//支持多級泛型的解析
for (int i = classes.length-1; i >0 ; i--) {
beforeType = new ParameterizedTypeImpl(new Type[]{beforeType == null? classes[i]:beforeType}, null, classes[i - 1]);
}
}
return JSON.parseObject(result,beforeType);
}
代碼評析:調試
主要起做用的仍是這兩行代碼rest
for (int i = classes.length-1; i >0 ; i--) {
beforeType = new ParameterizedTypeImpl(new Type[]{beforeType == null? classes[i]:beforeType}, null, classes[i - 1]);
}
主要意思是將你的class對象包裝成一個ParameterizedTypeImpl,使用ParameterizedTypeImpl來解析多級泛型,可是要注意的是,每層泛型,都須要用一個ParameterizedTypeImpl對象進行包裝起來纔會起做用,由於他會有一個actualTypeArguments
和一個 rawType
,在多級泛型中,用我這裏的例子說明,第一層的時候rawType
會是一個Result
對象,而actualTypeArguments
會是一個包裝事後的ParameterizedTypeImpl
對象,第二層的時候,rawType
會是一個List對象,而actualTypeArguments
會是包裝上一層的對象。後同。PS : 若是這裏不知道我說的是什麼,請調試的時候結合來看哈
最後也能夠正確解析哈~
END