• Retrofit解密:接口请求是如何适配suspend协程?


    最初的retrofit请求

    我们先看下原来如何通过retrofit发起一个网络请求的,这里我们直接以官网的例子举例:

    动态代理创建请求服务

    1. interface GitHubService {
    2.     //创建get请求方法
    3.     @GET("users/{user}/repos")
    4.     fun listRepos(@Path("user") user: String?): Call
    5. }
    6. //动态代理创建GitHubService
    7. fun createService(): GitHubService {
    8.     val retrofit = Retrofit.Builder()
    9.         .baseUrl("https://api.github.com/")
    10.         .build()
    11.     return retrofit.create(GitHubService::class.java)
    12. }
    • retrofit.create底层是通过动态代理创建的GitHubService的一个子实现类;

    • 创建的这个GitHubService一般作为单例进行使用,这里只是简单举例没有实现单例;

    发起网络请求

    1. fun main() {
    2.     //异步执行网络请求
    3.     createService().listRepos("").enqueue(object : Callback {
    4.         override fun onResponse(call: Call<Response>, response: retrofit2.Response<Response>) {
    5.             //主线程网络请求成功回调
    6.         }
    7.         override fun onFailure(call: Call<Response>, t: Throwable) {
    8.             //主线程网络请求失败回调
    9.         }
    10.     })
    11. }

    这种调用enqueue()异步方法并执行callback的方式是不是感觉很麻烦,如果有下一个请求依赖上一个请求的执行结果,那就将会形成回调地狱这种可怕场景。
    而协程suspend本就有着以同步代码编写执行异步操作的能力,所以天然是解决回调地狱好帮手。接下来我们看下如何使用协程suspend。

    借助suspend发起网络请求

    suspend声明接口方法

    1. interface GitHubService {
    2.     @GET("users/{user}/repos")
    3.     suspend fun listRepos(@Path("user") user: String?): Response
    4. }

    可以看到就是在listRepos方法声明前加了个suspend关键字就完了。

    创建协程执行网络请求

    1. fun main() {
    2.     //1.创建协程作用域,需要保证协程的调度器是分发到主线程执行
    3.     val scope = MainScope()
    4.     scope.launch(CoroutineExceptionHandler { _, _ ->
    5.         //2.捕捉请求异常
    6.     }) {
    7.         //3.异步执行网络请求
    8.         val result = createService().listRepos("")
    9.         val content = result.body()?
    10.     }
    11. }
    1. 首先创建一个协程作用域,需要保证协程调度器类型为Dispatchers.Main,这样整个协程的代码块都会默认在主线程中执行,我们就可以直接在里面执行UI相关操作

    2. 创建一个CoroutineExceptionHandler捕捉协程执行过程中出现的异常,这个捕捉异常的粒度比较大,是捕捉整个协程块的异常,可以考虑使用try-catch专门捕获网络请求执行的异常:

    1. //异步执行网络请求
    2. try {
    3.     val result = createService().listRepos("")
    4. catch (eException) {
    5.     //可以考虑执行重连等逻辑或者释放资源
    6. }
    1. 直接调用listRepos()方法即可,不需要传入任何回调,并直接返回方法结果。这样我们就实现了以同步的代码实现了异步网络请求。

    接下来我们就看下如何retrofit源码是如何实现这一效果的。

    retrofit如何适配suspend

    直接定位到HttpServiceMethod.parseAnnotations()方法:

    1. static <ResponseT, ReturnT> HttpServiceMethod<ResponseT, ReturnT> parseAnnotations(
    2.     Retrofit retrofit, Method method, RequestFactory requestFactory) {
    3.   //1.判断是否为suspend挂起方法
    4.   boolean isKotlinSuspendFunction = requestFactory.isKotlinSuspendFunction;
    5.   //省略一堆和当前分析主题不想关的代码
    6.   if (!isKotlinSuspendFunction) {
    7.     return new CallAdapted<>(requestFactory, callFactory, responseConverter, callAdapter);
    8.   } else if (continuationWantsResponse) {
    9.     //挂起执行
    10.     return (HttpServiceMethod<ResponseT, ReturnT>) new SuspendForResponse<>();
    11.   } else {
    12.     //挂起执行
    13.     return (HttpServiceMethod<ResponseT, ReturnT>) new SuspendForBody<>();
    14.   }
    15. }

    1.判断是否为suspend挂起方法

    看下requestFactory.isKotlinSuspendFunction赋值的地方,经过一番查找(省略…),最终方法在RequestFactory的parseParameter间接赋值:

    1. private @Nullable ParameterHandler<?> parseParameter() {
    2. //...
    3.     //1.是否是方法最后一个参数
    4.     if (allowContinuation) {
    5.       try {
    6.         if (Utils.getRawType(parameterType) == Continuation.class) {
    7.           //2.标识为suspend挂起方法
    8.           isKotlinSuspendFunction = true;
    9.           return null;
    10.         }
    11.       } catch (NoClassDefFoundError ignored) {
    12.       }
    13.     }
    14. }

    如果一个方法被声明为suspend,该方法翻译成java代码就会给该方法添加一个Continuation类型的参数,并且放到方法参数的最后一个位置,比如:

    1. private suspend fun test66(name: String) {  
    2. }

    会被翻译成:

    1. private final Object test66(String name, Continuation $completion) {
    2.    return Unit.INSTANCE;
    3. }

    所以上面的代码就可以判断出请求的接口方法是否被suspend声明,是isKotlinSuspendFunction将会被置为true

    2.挂起则创建SuspendForResponse或SuspendForBody

    这个地方我们以SuspendForBody进行分析,最终会执行到其adapt()方法:

    1. @Override
    2. protected Object adapt(Call<ResponseT> callObject[] args) {
    3.   call = callAdapter.adapt(call);
    4.   //1.获取参数
    5.   Continuation<ResponseT> continuation = (Continuation<ResponseT>) args[args.length - 1];
    6.   try {
    7.     return isNullable
    8.         ? KotlinExtensions.awaitNullable(call, continuation)
    9.         //2.调用真正的挂起方法
    10.         : KotlinExtensions.await(call, continuation);
    11.   } catch (Exception e) {
    12.     return KotlinExtensions.suspendAndThrow(e, continuation);
    13.   }
    14. }
    1. 获取调用的suspend声明的接口方法中获取最后一个Continuation类型参数

    2. 调用await方法,由于这是一个kotlin定义的接收者为Call的挂起方法,如果在java中调用,首先第一个参数要传入接收者,也就是call,其实await()是一个挂起方法,翻译成java还会增加一个Continuation类型参数,所以调用await()还要传入第一步获取的Continuation类型参数。

    3.核心调用await()方法探究

    await()就是retrofit适配suspend实现同步代码写异步请求的关键,也是消除回调地狱的关键:

    1. suspend fun  Call.await(): T {
    2.   return suspendCancellableCoroutine { continuation ->
    3.     continuation.invokeOnCancellation {
    4.       cancel()
    5.     }
    6.     enqueue(object : Callback {
    7.       override fun onResponse(call: Call<T>, response: Response<T>) {
    8.         if (response.isSuccessful) {
    9.           val body = response.body()
    10.           if (body == null) {
    11.               //关键
    12.             continuation.resumeWithException(KotlinNullPointerException())
    13.           } else {
    14.               //关键
    15.             continuation.resume(body)
    16.           }
    17.         } else {
    18.             //关键
    19.           continuation.resumeWithException(HttpException(response))
    20.         }
    21.       }
    22.       override fun onFailure(call: Call<T>, t: Throwable) {
    23.           //关键
    24.         continuation.resumeWithException(t)
    25.       }
    26.     })
    27.   }
    28. }

    使用到了协程的一个非常关键的方法suspendCancellableCoroutine{},该方法就是用来捕获传入的Continuation并决定什么恢复挂起的协程执行的,比如官方的delay()方法也是借助该方法实现的。

    所以当我们执行调用enqueue()方法时在网络请求没有响应(成功或失败)前,协程一直处于挂起的状态,之后收到网络响应后,才会调用resume()或resumeWithException()恢复挂起协程的执行,这样我们就实现了同步代码实现异步请求的操作,而不需要任何的callback嵌套地狱。

    总结

    本篇文章详细分析retrofit如何适配suspend协程的,并且不用编写任何的callback回调,直接以同步代码编写实现异步请求的操作。

  • 相关阅读:
    ORACLE-SQL 关于树结构的查询
    CSRF攻击原理详解
    410. 分割数组的最大值
    6.webpack4打包图片资源
    中国棉纺织行业市场深度分析及发展规划咨询综合研究报告
    ISL1208时钟芯片 Linux下 i2c 设置报警时钟。
    提权扫描工具
    STM32CubeMX学习笔记19——SD卡(SDIO接口)
    IPv4/IPv6、DHCP、网关、路由
    Azure KeyVault(四)另类在 .NET Core 上操作 Secrets 的类库方法-----Azure.Security.KeyVault.Secrets
  • 原文地址:https://blog.csdn.net/m0_73311735/article/details/127668418