Retrofit自定义解析器

作者: JuliusL | 来源:发表于2016-07-19 14:05 被阅读1216次

某些情况下,从服务器返回过来的json无法自动解析,只能通过手动解析(比如json某些字段的key是不固定的),如果项目中又正在用Retrofit作为网络库的话,就会比较头疼,因为Retrofit一般都只需要调用.addConverterFactory()传一个GsonConverterFactory对象就可以用Gson完成自动解析。然而网络上对于Retrofit的资源相当有限,摸索了一段时间终于知道如何用Retrofit自定义解析器了,写下来做个备忘,也为了方便后来人。

通常情况下Retrofit的代码会是这样:

private static OkHttpClient.Builder sOkHttpClient = new OkHttpClient.Builder();
private static Converter.Factory sGsonConverterFactory = GsonConverterFactory.create();
private static CallAdapter.Factory sRxJavaCallAdapterFactory = RxJavaCallAdapterFactory.create();

public static HotApi getHotApi(){
    if(sHotApi == null){
        Retrofit retrofit = new Retrofit.Builder()
                .client(sOkHttpClient.build())
                .baseUrl(BASE_URL)
                .addConverterFactory(sGsonConverterFactory)//这里一般传个GsonConverterFactory对象就可以用gson完成自动解析
                .addCallAdapterFactory(sRxJavaCallAdapterFactory)
                .build();
        sHotApi = retrofit.create(HotApi.class);
    }
    return sHotApi;
}

但是如果不想用Gson完成自动解析则需要自定义解析器了,总共分为四步。
步骤一:

public abstract class BaseResponseConverter<T> implements Converter<ResponseBody,T> {

    @Override
    public T convert(ResponseBody value) throws IOException {
        return parserJson(value.string());
    }

    public abstract T parserJson(String json);
}

public abstract class BaseConverterFactory<T> extends Converter.Factory {
    @Override
    public Converter<ResponseBody, T> responseBodyConverter(Type type, Annotation[] annotations, Retrofit retrofit) {
        return responseConverter();
    }

    public abstract BaseResponseConverter<T> responseConverter();

    @Override
    public Converter<?, RequestBody> requestBodyConverter(Type type, Annotation[] parameterAnnotations, Annotation[] methodAnnotations, Retrofit retrofit) {
        return super.requestBodyConverter(type, parameterAnnotations, methodAnnotations, retrofit);
    }
}

步骤二:

public class HotResponseConverter extends BaseResponseConverter<HotListBean>{

    @Override
    public HotListBean parserJson(String json) {
        HotListBean bean = new HotListBean();
        try {
            JSONObject jsonRoot = new JSONObject(json);
            String status = jsonRoot.getString("status");
            bean.setStatus(status);
            if ("ok".equals(status)) {
                JSONObject jsonPkg = jsonRoot.getJSONObject("list");
                bean.setId(jsonPkg.getString("id"));
                bean.setName(jsonPkg.getString("name"));
                bean.setType(jsonPkg.getString("type"));

                JSONArray JAcontents = jsonPkg.getJSONArray("contents");
                List<String> contents = new ArrayList<>();
                for (int i = 0; i < JAcontents.length(); i++) {
                    contents.add(JAcontents.getString(i));
                }
                bean.setContents(contents);

                bean.setTimestamp(jsonPkg.getLong("timestamp"));
                bean.setCount(jsonPkg.getInt("count"));

                JSONObject jsonDetails = jsonPkg.getJSONObject("details");
                List<EmoticonBean> details = new ArrayList<>();
                for (String emoId:contents) {
                    details.add(new Gson().fromJson(jsonDetails.getJSONObject(emoId).toString(), EmoticonBean.class));
                }
                bean.setDetails(details);
            }
        } catch (JSONException e) {
            e.printStackTrace();
        }
        return bean;
    }
}

步骤三:

public interface HotApi {
    @GET("v2_5/lists/hot")
    Observable<HotListBean> getHots(@Query("auth_token") String authToken,@Query("user_id")String userId,@Query("limit")int limit,@Query("page") int page);
}

步骤四:

public static HotApi getHotApi(){
    if(sHotApi == null){
        Retrofit retrofit = new Retrofit.Builder()
                .client(sOkHttpClient.build())
                .baseUrl(BASE_URL)
                .addConverterFactory(new BaseConverterFactory() {
                    @Override
                    public BaseResponseConverter responseConverter() {
                        return new HotResponseConverter();
                    }
                })
                .addCallAdapterFactory(sRxJavaCallAdapterFactory)
                .build();
        sHotApi = retrofit.create(HotApi.class);
    }
    return sHotApi;
}

相关文章

网友评论

  • b85958901227:谢谢大佬
  • f14b21ff662c:好文,简单明了实用:+1: 谢楼主分享
  • 勤知:在github上也看了下。没找到
  • 勤知:请问下BaseResponseConverter这个类是怎么写的,原谅我的愚昧。
    JuliusL:@勤知 谢谢指出,这是我遗漏了
    JuliusL:@勤知 已经补充到步骤一当中了

本文标题:Retrofit自定义解析器

本文链接:https://www.haomeiwen.com/subject/lioxjttx.html