使用Kotlin+协程+Flow+Retrofit+OkHttp搭建一套网络请求工具

本文主要是介绍使用Kotlin+协程+Flow+Retrofit+OkHttp搭建一套网络请求工具,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 一、前言
    • 二、搭建Java工程
      • 1.打开Android Studio 建一个Android 工程
      • 2.修改该工程的build.gradle文件为Java 工程,并配置如下:
      • 3. 搭建一个 RetrofitUtils 请求工具类,这都是常规操作了,如下
      • 4.建一个 NetWorkAPi 作为 Retrofit 请求接口:
      • 5.建一个 WXRepository 为了和Android 里面保持一致,可以直接拿过去用
      • 6.新建一个 WXMainTool.kt 类作为 主程序调用main 函数入口:
    • 三、运行调用接口

一、前言

  1. 怎么快速找调试接口?怎么快速找一个网络请求工具?
    很多人马上想到Retrofit + Okhttp 早起还有 HttpURLConnection 和 Volley但是 这大家常用的都是集成在Android程序里面,需 要安装运行在android设备里面,大多数需要触发按钮点击 才能调用访问接口,我们大多数情况下是需要 先看到数据结构正常,符合要求了,才进行开始写网络接口部分,特别是在自己研究某些网络接口,而这时候又没有类似可以调试的文档的时候

  2. 网络请求工具PC版上有好多工具。类似 Swagger 文档 , ApiFox ,或者其他在线网页请求工具?为什么又不用
    其实这个问题是,有类似网络调试接口文档是最好的,有时候没有的情况下或者说这些工具,大大小小都不是能完全满足某些需求,特别是 post的各种方式,或者put等不常用的请求方式

  3. 那 Retrofit + Okhttp 解耦的这么好用,可以搭建起来在PC上直接运行吗?可以的,不用那么麻烦的要查看一个接口数据,需要写好运行安装到手机里面,再点击按钮,这个过程大概需要 一分钟左右,如果写好代码直接调用,随时可以调用,不是节省很多时间。

  4. 这样做有什么好处?
    (1) 方便调试,方便快速查看接口数据,节省很多时间
    (2) 可以快速用PC写好的网络部分拷贝到Android工程项目里面用
    (3) 在没有调试接口文档,调用第三方Api或者研究其他接口时快速调用下接口后看内容
    (4) 网络抓包,找其他资源情况下,可以快速抓取到

二、搭建Java工程

1.打开Android Studio 建一个Android 工程

2.修改该工程的build.gradle文件为Java 工程,并配置如下:

plugins {id 'java-library'id 'kotlin'
}java {sourceCompatibility = JavaVersion.VERSION_1_8targetCompatibility = JavaVersion.VERSION_1_8
}dependencies {implementation "org.jetbrains.kotlin:kotlin-stdlib:1.7.0"implementation 'com.squareup.retrofit2:retrofit:2.9.0'implementation 'com.squareup.retrofit2:converter-scalars:2.9.0'implementation 'com.squareup.okhttp3:logging-interceptor:4.9.3'implementation 'com.squareup.okhttp3:okhttp:4.9.3'implementation 'com.squareup.okio:okio:2.10.0'implementation 'com.squareup.retrofit2:converter-gson:2.9.0'implementation 'com.google.protobuf:protobuf-java:3.5.1'implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.4.2'
}

3. 搭建一个 RetrofitUtils 请求工具类,这都是常规操作了,如下

class RetrofitUtils private constructor(private val baseUrl: String) {companion object {var instance: RetrofitUtils? = nullfun getInstance(baseUrl: String) = instance ?: synchronized(this) {instance ?: RetrofitUtils(baseUrl).also { instance = it }}}private inline val retrofit: Retrofitget() {val logging = HttpLoggingInterceptor()val timeout = 30000Lval okHttpClient = OkHttpClient.Builder().addInterceptor(HeaderInterceptor()).addInterceptor(logging)
//                .addInterceptor(RetrofitClient.BaseUrlInterceptor()).callTimeout(timeout, TimeUnit.MILLISECONDS)//设置连接超时.connectTimeout(timeout, TimeUnit.MILLISECONDS)//设置从主机读信息超时.readTimeout(timeout, TimeUnit.MILLISECONDS)//设置写信息超时.writeTimeout(timeout, TimeUnit.MILLISECONDS).retryOnConnectionFailure(true)//设置出现错误进行重新连接。.build();return Retrofit.Builder().client(okHttpClient).addConverterFactory(ScalarsConverterFactory.create()).addConverterFactory(GsonConverterFactory.create()).baseUrl(baseUrl).build()}fun <T> create(service: Class<T>?): T {if (service == null) {throw RuntimeException("Api service is null!")}return retrofit.create(service)!!}
}

4.建一个 NetWorkAPi 作为 Retrofit 请求接口:

interface NetWorkAPi {//示例post 请求@FormUrlEncoded@POST("https://www.wanandroid.com/user/register")suspend fun register(@Field("username") username: String,@Field("password") password: String,@Field("repassword") repassword: String): String// 示例get 请求@GET("article/list/0/json")suspend fun getHomeList(): String// 示例get 请求2@GET("article/list/{path}/json")suspend fun getHomeList22(@Path("path") page: Int): HomeData/************************* 以下只 示例写法,接口调不通,因为找不到那么多 公开接口  全是 Retrofit的用法 来测试 *****************************************************///    @FormUrlEncoded@Headers("Content-Type: application/x-www-form-urlencoded")  //todo 固定 header@POST("https://xxxxxxx")fun post1(@Body body: RequestBody): String//    @FormUrlEncoded@Headers("Content-Type: application/x-www-form-urlencoded")@POST("https://xxxxxxx22222")fun post12(@Body body: RequestBody, @HeaderMap map: Map<String, String>): String //todo  HeaderMap 多个请求头部自己填写
}

5.建一个 WXRepository 为了和Android 里面保持一致,可以直接拿过去用

class WXRepository private constructor() {companion object {val instance by lazy { WXRepository() }}private val baseUrl = "https://www.wanandroid.com/"private val api = RetrofitUtils.getInstance(baseUrl).create(NetWorkAPi::class.java)fun getHomeList() {runBlocking {// 主协程下面执行 flow 异步请求flow {emit(api.getHomeList())}.flowOn(Dispatchers.IO).catch {println(it.parseErrorString()) // 打印异常信息}.collect {println("网络返回结果:$it") // 打印异常信息}delay(10000) // 让程序主进程 在打印前不结束}}fun getHomeList22() {runBlocking {// 主协程下面执行 flow 异步请求flow {emit(api.getHomeList22(1))}.flowOn(Dispatchers.IO).catch {println(it)println(it.parseErrorString()) // 打印异常信息}.collect {it.data?.datas?.forEachIndexed { i, it ->println("第$i 条 : ${it.title}") // 打印异常信息}}delay(10000) // 让java工程 程序主进程 在打印前不结束}}fun register(username: String, password: String, repassword: String) {runBlocking {flow {emit(api.register(username, password, repassword))}.flowOn(Dispatchers.IO).catch {println(it)println(it.parseErrorString()) // 打印异常信息}.collect {println(it)}}}/************************* 以下只 示例写法,接口调不通,因为找不到那么多 公开接口  全是 Retrofit的用法 来测试 *****************************************************/fun post1() {runBlocking {flow {val map = mutableMapOf<String, Any>()map["AAAAAA"] = "AAAAAA"map["BBBBBB"] = 1map["CCCCCC"] = truemap["DDDDDD"] = 30.00femit(api.post1(RequestBodyWrapper(Gson().toJson(map))))}.flowOn(Dispatchers.IO).catch {println(it)println(it.parseErrorString()) // 打印异常信息}.collect {println(it)}}}fun post12() {runBlocking {flow {val map = mutableMapOf<String, Any>()map["AAAAAA"] = "AAAAAA"map["BBBBBB"] = 1map["CCCCCC"] = truemap["DDDDDD"] = 30.00fval headerMap = mapOf("Cookie" to "uab_collina=171755326751237298573402; tfstk=fFqEGyx3iMIEHdYy8fir_5GQozgKVDffUuGSE82odXcnd0awZ5V9FXiIAPlzs5lhp6CLa3Van4c7V09LWSwDAk_KV3-K20ffGisXpJn-qHga3LOpSxMhVYcnqYuGorffGis_pJn-qsa52bcgOYhMKYc3Zh0i6Y8oqX0ksFDmsbmoq7xGjxM2xU0oKdc9M402bY3h9NU_OKjk2plbKf-2BPkMqjsxT3Y7bvRjZJYDq3qZL2ESMIWRh2auHqEQQg-tA-zijYrRnUlruzqIjkfGmV3uz7DTW_8rgz2QkAqcFEh_bXNuYVXX04igcqm35NCIPDkoxqUOeeGur-rqAW19qXa4I5ubfCtjmrFa4qmc4XO-ISE82yRkr2xSQj6NI96RN__qDt2vyU3B6AlfK9YkrIKXVer5pUL-RaDZG9XA.; uag=7aed8d4dcb46b3eaa0e85f86b21df22e; ylogin=4024975; folder_id_c=10376791; phpdisk_info=WGhSYARkBz8BOgRjDmEEV1QwAAsAaAVlAzcDYw8xCjlTZFdjUjcDPFJiUAkMYFU9VD0NP109BWIPbgQxATMKPVg5UjUEMgdtAWAENQ5jBG5UZgA6AGoFawM4A2oPPwptU2FXYlJiAzxSZ1BhDF9VPlQzDThdNAVqDz0EZwExCj9Yb1Ji; PHPSESSID=pbba5e4111h3fk7om64mdgf4sbd0c6lm; __tins__21412745=%7B%22sid%22%3A%201717724688543%2C%20%22vd%22%3A%2012%2C%20%22expires%22%3A%201717727331949%7D; __51cke__=; __51laig__=12","Host" to "up.SSSSSwxxxoozooo.com","Origin" to "http://AAAAAAAAup.wxssosdsozosdoo.com","Referer" to "http://AAAAAAAup.wodsdsdsdsdosddszooo.com/mydisk.php?item=files&action=index&u=4024975")emit(api.post12(RequestBodyWrapper(Gson().toJson(map)), headerMap))}.flowOn(Dispatchers.IO).catch {println(it)println(it.parseErrorString()) // 打印异常信息}.collect {println(it)}}}
}

6.新建一个 WXMainTool.kt 类作为 主程序调用main 函数入口:

fun main() {
//    WXRepository.instance.getHomeList()WXRepository.instance.getHomeList22()
//    WXRepository.instance.register("WXXXXXXXXXX","1212121","1212121")}

三、运行调用接口

这里运行 WXRepository.instance.getHomeList22() ,而调用 WXRepository中的

fun getHomeList22() {runBlocking {// 主协程下面执行 flow 异步请求flow {emit(api.getHomeList22(1))}.flowOn(Dispatchers.IO).catch {println(it)println(it.parseErrorString()) // 打印异常信息}.collect {it.data?.datas?.forEachIndexed { i, it ->println("第$i 条 : ${it.title}") // 打印异常信息}}delay(10000) // 让java工程 程序主进程 在打印前不结束}
}

这篇关于使用Kotlin+协程+Flow+Retrofit+OkHttp搭建一套网络请求工具的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1114906

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

vite搭建vue3项目的搭建步骤

《vite搭建vue3项目的搭建步骤》本文主要介绍了vite搭建vue3项目的搭建步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1.确保Nodejs环境2.使用vite-cli工具3.进入项目安装依赖1.确保Nodejs环境

Nginx搭建前端本地预览环境的完整步骤教学

《Nginx搭建前端本地预览环境的完整步骤教学》这篇文章主要为大家详细介绍了Nginx搭建前端本地预览环境的完整步骤教学,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录项目目录结构核心配置文件:nginx.conf脚本化操作:nginx.shnpm 脚本集成总结:对前端的意义很多

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash