IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 移动开发 -> 从0开始封装一套项目的网络请求框架 -> 正文阅读

[移动开发]从0开始封装一套项目的网络请求框架

前言

Android进行网络请求,一般是通过Retrofit配合RxJava来实现。需要注意的是,在项目中,是不可能直接在每一个请求的地方直接裸着用框架,而是一定在自己的项目中对框架封装了一层,实际使用的是封装的这一层。
本篇文章就来介绍一种封装方案,在项目中可以使用。此方案肯定有不足的地方,还请批评指正。

用到的知识

Kotlin、Retrofit,RxJava,OkHttp

步骤介绍

1.api介绍

在这里使用鸿洋大神的开放api,链接点这里。只使用登录api即可,其他的api都是一样的方式。

url:https://www.wanandroid.com/user/login
方法:POST
参数:username,password

这里推荐一款很好用的接口测试工具Apipost,我这边测试一下登录的接口
在这里插入图片描述
它返回的response由三部分组成,分别是data,errorCode,errorMsg。我把结果粘贴过来

{
	"data": {
		"admin": false,
		"chapterTops": [],
		"coinCount": 10,
		"collectIds": [],
		"email": "",
		"icon": "",
		"id": 130077,
		"nickname": "LJH111",
		"password": "",
		"publicName": "LJH111",
		"token": "",
		"type": 0,
		"username": "LJH111"
	},
	"errorCode": 0,
	"errorMsg": ""
}
2.创建项目

创建一个kotlin项目,完成必要的准备
①依赖导入
在app模块的gradle文件中导入retrofit和rxjava的依赖
在这里插入图片描述

//导入retrofit
implementation 'com.squareup.retrofit2:retrofit:2.3.0'
implementation 'com.squareup.retrofit2:converter-gson:2.3.0'//ConverterFactory的Gson依赖包
implementation 'com.squareup.retrofit2:converter-scalars:2.3.0'//ConverterFactory的String依赖包
implementation("com.squareup.okhttp3:okhttp:4.9.1")
implementation 'com.squareup.retrofit2:adapter-rxjava2:2.3.0'

//导入rxjava
compile 'io.reactivex.rxjava2:rxandroid:2.0.1'
compile "io.reactivex.rxjava2:rxjava:2.0.8"

②开启网络权限
(1)首先新建一个xml文件,如图
在这里插入图片描述
内容是

<?xml version="1.0" encoding="utf-8"?>
<network-security-config>
    <base-config cleartextTrafficPermitted="true" />
</network-security-config>

(2)然后找到Menifest文件
在application标签中设置这样的属性
在这里插入图片描述
(3)然后,在menifest中打开网络权限
在这里插入图片描述
内容为

<!--网络权限-->
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
3.新建response模板类

所有的response都有一个共性,即由三部分组成,分别是data,errorCode,errorMsg。这里只有data的类型不确定,但是可以用泛型表示。
所以可以用一个通用的模板类来接收所有的Response

/**
 * Created by didiwei on 2022/4/20
 * desc: 所有请求的模板类
 */
data class ResponseModel<T>(val data: T,
                         val errorCode: String?,
                         val errorMsg: String?)

data用泛型表示即可

4.新建登录成功时,返回的data模板类

登录成功时,返回的json如下

{
	"data": {
		"admin": false,
		"chapterTops": [],
		"coinCount": 10,
		"collectIds": [],
		"email": "",
		"icon": "",
		"id": 130077,
		"nickname": "LJH111",
		"password": "",
		"publicName": "LJH111",
		"token": "",
		"type": 0,
		"username": "LJH111"
	},
	"errorCode": 0,
	"errorMsg": ""
}

其中因为data是不同的api,可能会不一样,所以最好用一个类来区分不同的data。这里登录成功时,返回的data我这样表示

/**
 * Created by didiwei on 2022/4/20
 * desc: 登录成功时,返回的response中的data的模板类
 */
data class LoginSuccessResData(val admin: Boolean,
                               val chapterTops: List<*>,
                               val coinCount: Int,
                               val collectIds: List<*>,
                               val email: String?,
                               val icon: String?,
                               val id: String?,
                               val nickname: String?,
                               val password: String?,
                               val publicName: String?,
                               val token: String?,
                               val type: Int,
                               val username: String?)
5.自定义RxJava操作符

正常的逻辑应该是这样的,请求成功时,把data丢给UI,请求失败时,把errorMsg丢给UI。所以我们可以自定义RxJava操作符,当数据返回时,对Response完成这样的功能。

/**
 * Created by didiwei on 2022/4/20
 * desc: RxJava的自定义操作符,用于剥离data和errorMsg
 */
abstract class RxJavaInterceptor<T> : Observer<ResponseModel<T>> {
    abstract fun success(data: T?)
    abstract fun failure(errorMsg: String?)

    override fun onSubscribe(d: Disposable?) {
        //这里可以弄个加载dialog的start,别忘了在请求结束(不管成功还是失败)的时候让这个dialog消失
    }

    override fun onNext(t: ResponseModel<T>?) {
        if(t?.data != null){
            //请求成功了
            success(t?.data)
        }else{
            //失败
            failure(t?.errorMsg)
        }
    }

    override fun onError(e: Throwable?) {
        failure(e?.message)
    }

    override fun onComplete() {
        //可以加个dialog的消失
    }

}

有个注意点,首先RxJavaInterceptor类为抽象类,里面的successfailure为抽象方法,当调用的时候实现。其次是这里依然是泛型类,适用于各个model

6.写请求的接口
/**
 * Created by didiwei on 2022/4/20
 * desc: 关于用户登录 注册的api
 */
interface UserService {
    /**
     * desc 登录的接口
     *
     * url:https://www.wanandroid.com/user/login
     * 方法:POST
     * 参数:username,password
     */
    @POST("/user/login")
    @FormUrlEncoded
    fun loginAction(@Field("username")username: String?,
                    @Field("password")password: String?)
        :Observable<ResponseModel<LoginSuccessResData>>
}

注意它的返回值是Observable类型,作为被观察者,流向观察者(RxJava的知识)

7.创建实例化service的类
/**
 * Created by didiwei on 2022/4/20
 * desc: 此类是单例类,用于创建Service的实例
 */
class ApiClient {
    private object Holder{
        val instance = ApiClient()
    }

    companion object{
        val INSTANCE = Holder.instance
    }

    fun <T> createRetrofit(retrofitInterface: Class<T>) : T{
        //创建okhttp
        val mOkHttpClient = OkHttpClient().newBuilder()
            .readTimeout(10000, TimeUnit.SECONDS)//读取超时时间
            .connectTimeout(10000, TimeUnit.SECONDS)//连接超时时间
            .writeTimeout(10000, TimeUnit.SECONDS)//写出超时时间
            .build()

        //创建retrofit
        val retrofit: Retrofit = Retrofit.Builder()
            .baseUrl("https://www.wanandroid.com")
            .client(mOkHttpClient)
            .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
            .addConverterFactory(GsonConverterFactory.create())
            .build()
        
        //返回
        return retrofit.create(retrofitInterface)
    }
}
8.这样当我们使用的时候,就可以这样用

比如添加一个登录按钮,然后注册其点击事件

class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        val userLoginBtn: Button = findViewById<Button>(R.id.userLoginBtn)
        userLoginBtn.setOnClickListener(onClickListener)
    }

    private val onClickListener = View.OnClickListener {
        when(it.id){
            R.id.userLoginBtn ->{
                val userName = "LJH111"
                val userPwd = "123456"

                ApiClient.INSTANCE.createRetrofit(UserService::class.java)
                        .loginAction(userName,userPwd)
                        .subscribeOn(Schedulers.io())
                        .observeOn(AndroidSchedulers.mainThread())
                        .subscribe(object : RxJavaInterceptor<LoginSuccessResData>(){
                            override fun success(data: LoginSuccessResData?) {
                                Log.v("ljh","请求成功,返回的data为:${data}")
                            }

                            override fun failure(errorMsg: String?) {
                                Log.v("ljh","请求失败,错误信息为:${errorMsg}")
                            }
                        })
            }
        }
    }
}

效果
在这里插入图片描述
如果改成错误的用户名或密码,效果是这样的
在这里插入图片描述
至此,我们就完成了网络请求规则的封装

  移动开发 最新文章
Vue3装载axios和element-ui
android adb cmd
【xcode】Xcode常用快捷键与技巧
Android开发中的线程池使用
Java 和 Android 的 Base64
Android 测试文字编码格式
微信小程序支付
安卓权限记录
知乎之自动养号
【Android Jetpack】DataStore
上一篇文章      下一篇文章      查看所有文章
加:2022-04-23 10:58:30  更:2022-04-23 10:59:05 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 0:00:50-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码