Kotlin使用Retrofit调用后台接口(以登录为例)

1、创建进行网络请求的接口

interface Api {
    @POST("api/login")
    fun login(@Body request: LoginRequest): MyCall<LoginToken>
//返回值是Retrofit网络请求回来json数据,经过Gson转换以后的对象
//@Body会把参数放到请求体中,适用于POST请求。
}

POST注解
Retrofit在创建的时候,有一行代码:

.baseUrl(baseUrl)

这个baseUrl是我们要访问的接口的baseUrl,而我们现在用POST注解的字符串 "api/login"会追加到baseUrl后面,通过去调用这个接口得到返回数据。
2、创建Retrofit对象

class WebService private constructor() {
    companion object {
        var api: Api? = null
            get() {
                if (field == null) {
                    field = createApi(Api.BASE_URL)
                }
                return field
            }

        fun get(): Api {
            return api!!
        }

        private fun createApi(baseUrl: String): Api {
            val retrofit = Retrofit.Builder()
                    .client(createHttpClient())
                    .baseUrl(baseUrl)//设置网络请求的Url地址
                    .addConverterFactory(GsonConverterFactory.create())//设置数据解析器,使得来自接口的json结果会自动解析成定义好了的字段和类型都相符的json对象接受类
                    .addCallAdapterFactory(MyCallAdapterFactory())//想把返回值定义为Observable对象
                    .build()
            // 返回网络请求接口的实例
            return retrofit.create(Api::class.java)
        }
}

这里通过调用 WebService.get()的方式就可以获得Retrofit对象

3、发送网络请求

fun login(userName: String, password: String): LiveData<NetworkResource<LoginToken>> {
        val result = MutableLiveData<NetworkResource<LoginToken>>()//LiveData的子类,可进行setValue、postValue操作
        val request = LoginRequest()
        request.username = userName
        request.pwd = password
        WebService.get().login(request).enqueue(object : ApiCallback<LoginToken> {//创建网络请求接口的实例并对发送请求进行封装,然后enqueue发送网络请求
            //enqueue()非异步方式,会阻塞线程,等待返回结果。
            override fun onSuccess(response: LoginToken) {//成功时返回LoginTakon对象
                App.get().saveUserToken(response.token!!)//这句后面专门讲
                result.postValue(NetworkResource.success(response))//成功后返回的LoginToken的对象respose,放入MutableLiveData对象result中
            }

            override fun onError(error: HError) {
                result.postValue(NetworkResource.error(error, null))
            }
        })
        return result
    }

第三行的LoginRequest类:

class LoginRequest {
    var username: String? = null
    var pwd: String? = null
}

先从WebService那行开始讲,这里对Callback进行了一个封装
ApiCallback:

interface ApiCallback<T> {
    fun onSuccess(response: T)
    fun onError(error: HError)
}

object:Apicallback{.......}是个对象表达式,相当于java里new的一个匿名类
重写Apicallback<LoginToken>里的两个函数

4、LiveData与ViewModel的组合使用

LiveData:数据变化时会收到通知。
ViewModel:管理UI的数据

利用这一点,使UserViewModel的login函数的返回值为LiveData对象,当他的数据变化时,LoginActivity的“观察者”会观测到并触发相应的事件处理。

 if (userViewModel == null) {//为空创建UserViewModel对象,返回值是LiveData<LoginToken>对象
      userViewModel = ViewModelProviders.of(this).get(UserViewModel::class.java)
      //of(this),这里this代表LoginActivity,表示生命周期就以LoginActivity作为参照对象了。
      //get(UserViewModel::class.java),表示与UserViewModel关联
 }
 showLoadingDialog()//显示加载框
 userViewModel!!.login(name, pwd).observe(this, Observer { resource ->
//观测者发现LiveData对象发生变化时
 closeLoadingDialog()//关闭加载框
 if (resource!!.isSuccess) {
    startActivity(MainActivity::class.java)
    App.get().uploadFireBaseMsgToken()
    finish()
 } else {
    val msg = resources.getString(R.string.login_fail)
    Toaster.shortToast(msg!!)
 }

5、回到第3中

override fun onSuccess(response: LoginToken) {//成功时返回LoginTakon对象
      App.get().saveUserToken(response.token!!)
      result.postValue(NetworkResource.success(response))//成功后返回的LoginToken的对象respose,放入MutableLiveData对象result中
 }

App.get().saveUserToken(response.token!!)这行代码一看就知道是客户端存储了一个东西——服务端返回的LoginToken对象第token变量。
这里看一下LoginToken里面有什么

class LoginToken : BaseResponse() {
    var token: String? = null
}

LoginToken继承自BaseResponse,BaseResponse是自己定义的一个类,用于判断是否操作成功,错误类型...
可以看到LoginToken里面定义了一个变量token
那客户端为什么要存储服务端返回的这个token变量呢,这个token变量又是什么?

user_token的职责是保护用户的用户名及密码多次提交,以防密码泄露。
如果接口需要用户登录,其访问流程如下:
1、用户提交“用户名”和“密码”,实现登录
2、登录成功后,服务端返回一个user_token,生成规则参考如下:
user_token = md5('用户的uid' + 'Unix时间戳') = etye0fgkgk4ca2ttdsl0ae9a5dd77471fgf

总结:
服务端生成user_token后,返回给客户端(自己存储),这里的
App.get().saveUserToken(response.token!!)
就是将服务端生成的user_token存储到客户端,客户端每次接口请求时,如果接口需要用户登录才能访问,则需要把 user_id与user_token传回给服务端。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,406评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,732评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,711评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,380评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,432评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,301评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,145评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,008评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,443评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,649评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,795评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,501评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,119评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,731评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,865评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,899评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,724评论 2 354

推荐阅读更多精彩内容