Android Retrofit的源码解读
先提出几个问题,然后本片文章就根据下面几个问题做源码的解读。
1.整个请求的流程是怎样的?
2.底层是如何用 OkHttp 请求的?
3.Okhttp是异步的,retrofit是怎么帮我们切换到主线程的
4.注解是什么时候解析的,怎么解析的?
5.Converter和CallAdapter的作用
6.如何支持 Kotlin协程的suspend挂起函数的?
Retrofit的简单流程图
data:image/s3,"s3://crabby-images/1b591/1b5913055d7738c903a7e74aa53c2257dfd447a5" alt=""
接下来就根据流程图进入,首先是Retrofit.build()
data:image/s3,"s3://crabby-images/42044/42044d1b40c43ec1019c2ce7a6208785631f8512" alt=""
重点1
因为我们一般都不会传入callbackExecutor(回调执行器),所以他会进入到platform.defaultCallbackExecutor()语句
但是我们直接点进去看的话,会发现,啥也没有
data:image/s3,"s3://crabby-images/8df6c/8df6c9ad6b0e266e8c06b30ed0682b6a1fc86513" alt=""
那就先从platform入手吧,进入之后能看到
(其实英语也知道platform就是平台的意思,这里就是做一个平台的区分)
data:image/s3,"s3://crabby-images/107a6/107a608d0600a271c0432138810e93af2a86ffd1" alt=""
接着再进入Android()
data:image/s3,"s3://crabby-images/c13a7/c13a718273446e402dcf37e31957f4ccf7681d5c" alt=""
所以这里可以认为callbackExecutor == MainThreadExecutor
,用来切换线程
重点2
可以看到上一步拿到的callbackExecutor 作为参数传入了platform.defaultCallAdapterFactories(callbackExecutor)
,进入看到
data:image/s3,"s3://crabby-images/4ea68/4ea68cf104cfcc908ad49b5a792deff042af694f" alt=""
接着进入DefaultCallAdapterFactory
data:image/s3,"s3://crabby-images/bf538/bf538985ce7f3186412ed6e40da27d6bc3705c05" alt=""
在进入ExecutorCallbackCall
data:image/s3,"s3://crabby-images/bb34c/bb34ca67352e02c6302c3396f9243634fc4750d6" alt=""
这就是发起网络请求后,Retrofit帮我们切换到主线程的流程,总得来说就是在build
方法中,通过Platform
的子类Android()
添加了一个默认的Executor:MainThreadExecutor
,然后还添加了一个DefaultCallAdapterFactory
并创建了匿名内部类CallAdapter
将MainThreadExecutor
,包装成ExecutorCallbackCall
,在网络请求回调,通过它切换线程,并把结果抛出去
重点3
data:image/s3,"s3://crabby-images/70784/707843b86e87d2004d03cde9d9c82060f3901dd7" alt=""
如果你没添加ConverterFacTory的话,默认会通过这个转换数据,你也可以通过继承Converter.FacTory自定义自己的数据转换。Gson的就是这样做的。
接着Retrofit.create()
这个方法看起来很简单就是通过动态代理获取interface的实例,接着你就可以调用方法了
当我们通过实例调用方法时,就会进入InvocationHandler.invoke
,重点在这里面
data:image/s3,"s3://crabby-images/f1aec/f1aec3eb547045420cfb47d5e91608165cd9cc79" alt=""
先进入到看一看ServiceMethod
进入loadServiceMethod
,
方法很简单,就是先从自己保存的中查找,没有则生成再保存下来,以后可以复用。
data:image/s3,"s3://crabby-images/45b93/45b93cb51909c54b0fd76508fef47bdc6905d189" alt=""
data:image/s3,"s3://crabby-images/d1822/d1822c4a5179d813557a12e6923b831aea46f6d0" alt=""
先看一下RequestFactory
,里面都是方法注解的处理。需要注意的是对kotlin协程的适配
在RequestFactory的build方法中
,会遍历所有参数,去解析他们
data:image/s3,"s3://crabby-images/04459/04459e1ead8e29d6f71ac0dc06a1550fc9fc4309" alt=""
在跟进到parseParameter
,我们知道kotlin在编译时候,会给方法最后后面添加一个Continuation<T>
的参数,
data:image/s3,"s3://crabby-images/161b1/161b17cbd6b9ea589b5e66532ca52d82681cbecd" alt=""
而这个标志位会保存在requestFactory
然后传入HttpServiceMethod
!!这里写错了,这里responseType应该是Continuation里面的泛型<T>!!
data:image/s3,"s3://crabby-images/aae00/aae002523eff106b40d5a7d9fbf3e4ec61e6a26b" alt=""
data:image/s3,"s3://crabby-images/e11d0/e11d03a0d6c66d1e0dfc1fab14e476e37b1cf39c" alt=""
1.先看第一个,普通方法进入的CallAdapted
写错了,应该是loadServiceMethod(method).invoke()
调用的
data:image/s3,"s3://crabby-images/ba3cf/ba3cf0a4258cdec9b7672c2368b2bf1854b1ede9" alt=""
data:image/s3,"s3://crabby-images/9aaa2/9aaa279ddf58932591e4260bb728876cdaf9effd" alt=""
2.当是一个suspend方法时
data:image/s3,"s3://crabby-images/e4295/e4295e2fa27665657ccabf1fc3691b44c1f528db" alt=""
data:image/s3,"s3://crabby-images/ed64a/ed64a943b1dc41f17f0d215f68c0cc2b62284d02" alt=""
interface.enqueue
在调用apiService.getHttp() 接口的方法
其实就是在调用 --> InvocationHan.invoke()
最终就是 --> return loadServiceMethod(method).invoke()
在跟进代码之后,
data:image/s3,"s3://crabby-images/18900/18900a3608e2f04f454a5224abe29b10cd5f754b" alt=""
所以又回到了DafalutCallAdapterFactory
data:image/s3,"s3://crabby-images/dfa9d/dfa9d9d32d71521edcedc9b703235418cf1edbd9" alt=""
看到OkHttpCall的方法
data:image/s3,"s3://crabby-images/ef697/ef697fa3f0fd47889c5dc1bef14a2f1fc5761c37" alt=""
这是真正的创建OkHttp的Call
data:image/s3,"s3://crabby-images/b1770/b17700a3bf34f8cbcb93d63d6bb209cab55e968f" alt=""
拿到Okhttp的call后就可以网络请求了
data:image/s3,"s3://crabby-images/5eb09/5eb09e6aedc020ba46995e18a21c04c865797630" alt=""