【开源库学习】从OkHttp到Retrofit(其二 Retrofit)
创始人
2024-05-29 23:23:25
0

从OkHttp到Retrofit

  • 简单使用
  • 实现原理
    • loadServiceMethod
    • Converter

简单使用

class RetrofitActivity : AppCompatActivity() {companion object {const val SERVER = "https://www.xxx.com/"}var disposable:Disposable? = null;override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_retrofit)val retrofit = Retrofit.Builder().baseUrl(SERVER).addConverterFactory(GsonConverterFactory.create()).addCallAdapterFactory(RxJava2CallAdapterFactory.create()).build()disposable = retrofit.create(TestApi::class.java).getIcons(1).subscribe({Log.d("api", it.isSuccessful.toString())}, {Log.d("api", it.toString())})}override fun onStop() {super.onStop()disposable.dispose()}
}//RESTFUL 风格接口
interface TestApi {@GET("/test/icon")fun getIcons(@Query("icon_id") icon_id: Long): Observable>
}class IconBean{}

实现原理

retrofit
Retrofit的代码逻辑比较复杂,但是实际上Retrofit的主要功能就是给OkHttp做一层封装,这里我把主要流程进行了总结,简单来说当调用Retrofit.create(TestApi.classjava)方法之后,Retrofit首先会判断TestApi这个类是否符合条件,如果开启了严格检查的话,还会提前加载HttpServiceMethod。

//Retrofit.java
public  T create(final Class service) {//验证validateServiceInterface(service);return (T)//动态代理Proxy.newProxyInstance(service.getClassLoader(), new Class[] {service}, new InvocationHandler() {//判断android和jvm平台及其版本private final Platform platform = Platform.get();@Overridepublic Object invoke(Object proxy, Method method, Object[] args){//如果是静态方法则直接执行if (method.getDeclaringClass() == Object.class) {return method.invoke(this, args);}//isDefaultMethod:是否是默认方法return platform.isDefaultMethod(method)? platform.invokeDefaultMethod(method, service, proxy, args): loadServiceMethod(method).invoke(args); }});
}
private void validateServiceInterface(Class service) {//判断是不是接口,不是就抛出异常if (!service.isInterface()) {throw new IllegalArgumentException("API declarations must be interfaces.");}//判断存不存在泛型,存在则抛出异常Deque> check = new ArrayDeque<>(1);check.add(service);while (!check.isEmpty()) {Class candidate = check.removeFirst();if (candidate.getTypeParameters().length != 0) {StringBuilder message =new StringBuilder("Type parameters are unsupported on ").append(candidate.getName());if (candidate != service) {message.append(" which is an interface of ").append(service.getName());}throw new IllegalArgumentException(message.toString());}Collections.addAll(check, candidate.getInterfaces());}//提前加载if (validateEagerly) {Platform platform = Platform.get();for (Method method : service.getDeclaredMethods()) {if (!platform.isDefaultMethod(method) && !Modifier.isStatic(method.getModifiers())) {loadServiceMethod(method);}}}}

loadServiceMethod

当检查完成之后,Retrofit就会创建出一个代理类,因此调用retrofit.create(TestApi::class.java).getIcons(1)事实上是调用了代理类的invoke方法,这个invoke方法会首先根据注解进行解析,添加参数,生成一个okHttp的Call请求,然后通过适配器模式将Call的返回结果进行适配。例如我希望使用RxJava的Observable作为返回类型,那么这里可以使用RxJavaCallAdapterFactory将OkHttp返回的Call类型转换为Observable
基于这个流程,来阅读一下代码。

ServiceMethod loadServiceMethod(Method method) {//判断是否有缓存ServiceMethod result = serviceMethodCache.get(method);if (result != null) return result;synchronized (serviceMethodCache) {result = serviceMethodCache.get(method);if (result == null) {//没有缓存的话根据注解解析一个//这里的返回是一个HttpServiceMethod,里面包含了一个适配好的callresult = ServiceMethod.parseAnnotations(this, method);serviceMethodCache.put(method, result);}}return result;}
//ServiceMethod.java
static  ServiceMethod parseAnnotations(Retrofit retrofit, Method method) {RequestFactory requestFactory = RequestFactory.parseAnnotations(retrofit, method);//检查:articleList方法返回类型不能用通配符和void...return HttpServiceMethod.parseAnnotations(retrofit, method, requestFactory);
}

可以看到ServiceMethod中首先创建了一个RequestFactory,RequestFactory的作用是根据注解来

//RequestFactory.java
static RequestFactory parseAnnotations(Retrofit retrofit, Method method) {return new Builder(retrofit, method).build();
}class Builder {RequestFactory build() {//解析请求类型,methodAnnotations是method中的注解,通过getAnnotations()获取for (Annotation annotation : methodAnnotations) {parseMethodAnnotation(annotation);}//...//解析参数注解如Path,parameterAnnotationsArray是参数注解数组int parameterCount = parameterAnnotationsArray.length;parameterHandlers = new ParameterHandler[parameterCount];for (int p = 0, lastParameter = parameterCount - 1; p < parameterCount; p++) {parameterHandlers[p] =parseParameter(p, parameterTypes[p], parameterAnnotationsArray[p], p == lastParameter);}//... return new RequestFactory(this);}
}

从这里可以看出当build方法调用之后,事实上RequestFactory对传入的方法完成了解析,并且把解析的结果保存在自己成员变量中。

//HttpServiceMethod.java
//ResponseT响应类型如WanArticleBean,ReturnT返回类型如Call
static  HttpServiceMethod parseAnnotations(Retrofit retrofit, Method method, RequestFactory requestFactory) {//...Annotation[] annotations = method.getAnnotations();//遍历找到合适的适配器CallAdapter callAdapter =createCallAdapter(retrofit, method, adapterType, annotations);//得到响应类型Type responseType = callAdapter.responseType();//根据返回类型找到合适的转换器//最终又会回调到retrofit里面的callAdapterFactories//callAdapterFactories在一开始builder时候传入Converter responseConverter =createResponseConverter(retrofit, method, responseType);okhttp3.Call.Factory callFactory = retrofit.callFactory;return new CallAdapted<>(requestFactory, callFactory, responseConverter, callAdapter);
}static final class CallAdapted extends HttpServiceMethod {private final CallAdapter callAdapter;CallAdapted(RequestFactory requestFactory,okhttp3.Call.Factory callFactory,Converter responseConverter,CallAdapter callAdapter) {super(requestFactory, callFactory, responseConverter);this.callAdapter = callAdapter;}@Overrideprotected ReturnT adapt(Call call, Object[] args) {return callAdapter.adapt(call);}//父类HttpServiceMethod中的实现。@Overridefinal @Nullable ReturnT invoke(Object[] args) {Call call = new OkHttpCall<>(requestFactory, args, callFactory, responseConverter);return adapt(call, args);}}

结合之前的说明,可以看到当找到和是的适配器之后,loadServiceMethod会返回一个CallAdapted,并且当调用invoke方法时候,会根据CallAdapted内部的callAdapter,将Call转化为适配类型。
callAdapter又是从retrofit的callAdapterFactories中遍历得到,所以这里再回过来看看retrofit.builder()方法。

//Retrofit.Builder.java
public Retrofit build() {Executor callbackExecutor = this.callbackExecutor;//如果没设置线程池,则给android平台设置一个默认的MainThreadExecutor(用Handler将回调切回主线程)if (callbackExecutor == null) {callbackExecutor = platform.defaultCallbackExecutor();}List callAdapterFactories = new ArrayList<>(this.callAdapterFactories);//添加默认的DefaultCallAdapterFactorycallAdapterFactories.addAll(platform.defaultCallAdapterFactories(callbackExecutor));
}
List defaultCallAdapterFactories(@Nullable Executor callbackExecutor) {DefaultCallAdapterFactory executorFactory = new DefaultCallAdapterFactory(callbackExecutor);return hasJava8Types//默认的只有两个,如果需要使用其他的AdapterFactory需要在创建的时候传入? asList(CompletableFutureCallAdapterFactory.INSTANCE, executorFactory): singletonList(executorFactory);}

Converter

Converter的作用就是转化各种参数类型。

interface WanApi {//Long cur 当前时间@GET("article/list/{page}/json")Call articleList(@Path("page") int page, @Query("cur") Long cur);
}class TimeConverter implements Converter {private SimpleDateFormat mFormat = new SimpleDateFormat("yyyy-MM-dd-HHmmss");@Overridepublic String convert(Long value) throws IOException {if (value > 1_000_000_000_000L) {//毫秒,不是很严谨 - -return mFormat.format(new Date(value));}return String.valueOf(value);}
}class TimeConverterFactory extends Converter.Factory {@Overridepublic Converter stringConverter(Type type, Annotation[] annotations, Retrofit retrofit) {if (type == Long.class) {//使用自定义TimeConverterreturn new TimeConverter();}return super.stringConverter(type, annotations, retrofit);}public static Converter.Factory create() {return new TimeConverterFactory();}
}//addConverterFactory(TimeConverterFactory.create())

相关内容

热门资讯

pc掌机安装安卓系统,PC掌机... 你有没有想过,你的PC掌机也能装上安卓系统,变成一个全能的小神器呢?没错,今天就来给你详细介绍如何让...
苹果11像安卓系统,深度解析安... 你有没有发现,最近苹果11的操作系统越来越像安卓系统了呢?这可不是我随便说说哦,咱们来好好聊聊这个话...
安卓系统支付宝图标,简约而不简... 你有没有发现,手机屏幕上那个小小的支付宝图标,简直就像是个忠诚的小卫士,时刻守护着你的钱包呢?今天,...
梦幻q传没有安卓系统,探索全新... 你有没有听说最近的一款游戏《梦幻q传》?这款游戏可是让不少玩家为之疯狂呢!不过,你知道吗?它竟然没有...
TCL安卓系统电视刷机,解锁更... 亲爱的电视迷们,你是否对家里的TCL安卓系统电视感到有些厌倦,想要给它来个焕然一新的变化呢?那就跟着...
安卓系统后起之秀,揭秘后起之秀... 你知道吗?在手机操作系统这个江湖里,安卓系统可是个后起之秀呢!它就像一位默默无闻的侠客,突然间就闯出...
安卓转换电脑系统,安卓系统在电... 你有没有想过,你的安卓手机里的那些好玩的应用和游戏,能不能直接在电脑上畅玩呢?今天,就让我来带你一探...
安卓系统的爱心教程,轻松实现爱... 亲爱的安卓用户们,是不是觉得手机里的爱心图标有点神秘,不知道怎么设置呢?别急,今天就来给你来一场爱心...
适配苹果系统和安卓系统的手表,... 你有没有发现,最近手腕上的小玩意儿越来越受欢迎了呢?没错,说的就是那些集时尚与科技于一身的手表。今天...
安卓系统占用突然增高,揭秘原因... 最近你的安卓手机是不是突然感觉有点卡呢?是不是觉得内存不够用,打开个应用都要等半天?别急,今天就来给...
安卓系统怎样远程桌面windo... 你有没有想过,在安卓手机上也能轻松操控Windows系统呢?没错,这就是今天我要跟你分享的神奇技巧—...
安卓获取系统设置的铃声,解锁个... 你有没有想过,手机里那些悦耳的铃声,其实都是可以自己挑选的?没错,就是那个安卓系统里隐藏的小秘密!今...
安卓多屏操作系统,多维度交互体... 你有没有发现,最近手机屏幕越来越大,操作起来是不是有点儿手忙脚乱?别急,今天就来给你揭秘一个神奇的东...
安卓系统没有通话界面,探索安卓... 你有没有遇到过这种情况?手机屏幕上明明显示着通话界面,但是当你点进去的时候,却发现什么都没有?没错,...
安卓子系统没反应,安卓子系统故... 手机突然卡壳了,安卓子系统没反应,这可怎么办?别急,让我带你一步步排查,找出问题所在,让你的手机重焕...
安卓系统装树莓派,安卓系统下树... 你有没有想过,把安卓系统装到树莓派上,会是个什么样子呢?想象一个迷你的小树莓派,竟然能运行起我们熟悉...
海信操作系统和安卓,打造智能生... 你知道吗?在智能电视的世界里,操作系统可是个大角色呢!今天,咱们就来聊聊海信的操作系统和安卓,看看它...
召唤抽奖系统1.9安卓pc,惊... 你知道吗?最近网上掀起了一股热潮,那就是召唤抽奖系统1.9安卓PC版。这款游戏可是让不少玩家为之疯狂...
安卓如何导入鸿蒙系统文件,鸿蒙... 你有没有想过,让你的安卓手机也来个华丽变身,体验一下鸿蒙系统的魅力呢?别急,今天就来手把手教你如何导...
安卓系统播放sd卡音乐,轻松播... 你有没有遇到过这种情况:手机里存了满满的音乐,却因为系统的问题无法顺畅播放?别急,今天就来给你详细说...