Android实战 -> 使用Interceptor+Lock实现无缝刷新Token

前言

哈喽各位我又来了,相信大家在做APP的时候肯定会遇到用户Token即将过期或者已经过期的情况,那么这个时候后端都返回相应的Code提示我们要求刷新Token处理。

那么今天这篇文章就给大家提供一个思路。

开工

技术点
  • Interceptor -> 拦截器
  • ReentrantLock -> 重入锁
实现思路
  • 通过TokenInterceptor获取Response解析请求结果验证是否Token过期
  • 监控到Token已过期后阻塞当前线程,调用刷新Token接口并使用Lock锁
  • 并发的请求也监控到了Token过期后,先校验Lock是否已锁,已锁等待,未锁步骤2
  • Token刷新成功后各线程携带新的Token创建Request重新请求

总结:4个并发线程接口,谁抢到了Lock锁谁去刷新Token,其他三个线程阻塞等待

实现代码
kotlin 复制代码
private fun handle(name: String) {
    Log.d(TAG, "handle 【Start】 called with: name = $name")
    try {
        if (!mLock.isLocked) {
            this.mLock.lock() // 加锁
            Log.d(TAG, "handle 【Start  Refresh】 called with: name = $name")
            Thread.sleep(5000) // 此处应为刷新Token请求
            Log.d(TAG, "handle 【End Refresh】 called with: name = $name")
            this.mLock.unlock() // 释放锁
        } else {
            Log.d(TAG, "handle 【Wait Refresh】 called with: name = $name")
            while (true) { // 阻塞等待
                if (!mLock.isLocked) { // 查询锁状态
                    Log.d(TAG, "handle 【OK Refresh】 called with: name = $name")
                    break
                }
            }
        }
    } finally {
        if (mLock.isLocked) {
            this.mLock.unlock()
        }
    }
    Log.d(TAG, "handle 【End】 called with: name = $name")
}

如上述代码,抢到Lock锁的线程去刷新Token,其余线程等待结果。

模拟测试
arduino 复制代码
// 此处模拟并发请求
this.findViewById<View>(R.id.btnGo).setOnClickListener {
    thread {
        handle("线程1")
    }
    thread {
        handle("线程2")
    }
    thread {
        handle("线程3")
    }
}
输出日志

如图,线程2抢到了Lock锁,线程1、3则进入了等待状态

如图,线程2刷新Token成功后释放了锁,线程1、3监听到了锁被释放则进入重新请求逻辑

完整代码
kotlin 复制代码
class MainAc : AppCompatActivity() {

    private val TAG = "zzz"

    private val mLock by lazy { ReentrantLock() }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        super.setContentView(R.layout.ac_main)

        this.findViewById<View>(R.id.btnGo).setOnClickListener {
            thread {
                handle("线程1")
            }
            thread {
                handle("线程2")
            }
            thread {
                handle("线程3")
            }
        }
    }


    private fun handle(name: String) {
        Log.d(TAG, "handle 【Start】 called with: name = $name")
        try {
            if (!mLock.isLocked) {
                this.mLock.lock()
                Log.d(TAG, "handle 【Start  Refresh】 called with: name = $name")
                Thread.sleep(5000)
                Log.d(TAG, "handle 【End Refresh】 called with: name = $name")
                this.mLock.unlock()
            } else {
                Log.d(TAG, "handle 【Wait Refresh】 called with: name = $name")
                while (true) {
                    if (!mLock.isLocked) {
                        Log.d(TAG, "handle 【Go Refresh】 called with: name = $name")
                        break
                    }
                }
            }
        } finally {
            if (mLock.isLocked) {
                this.mLock.unlock()
            }
        }
        Log.d(TAG, "handle 【End】 called with: name = $name")
    }
}
End

到这里就结束了,简单吧,希望可以帮到在座的小伙伴们。当然如果有更好的实现方式或方案也希望各位在评论区留言讨论,我秒回复哦~ Bye

相关推荐
光影少年3 分钟前
大屏页面,一次多个请求,请求加密导致 点击 全局时间选择器 时出现卡顿咋解决(面板收起会延迟1~2秒)
前端·javascript·vue.js·学习·前端框架·echarts·reactjs
Mr.mjw18 分钟前
vue中封装一个环形进度条组件,根据外部盒子大小自适应变化
前端·javascript·vue.js
无心使然20 分钟前
Openlayers调用ArcGis影像服务之一动态地图、地图切片(/exportImage)
前端·javascript·数据可视化
唯火锅不可辜负27 分钟前
uniapp开发公众号订阅功能踩坑小记
前端·vue.js
opteOG1 小时前
游览器跨域问题详解
前端
SameX1 小时前
后台 GPS 记录从半天掉电 30% 到全天 8%,我的三版方案演进
前端
Cder1 小时前
用 React + Ink 在终端里「优雅搜索」:开源 CLI 设计与非交互模式实践
前端·agent
像我这样帅的人丶你还1 小时前
前端监控体系与实践(二):全局监控
前端·javascript·vue.js
颜酱1 小时前
LLM为核,上下文为限:拆解AI Agent生态的底层逻辑
前端·人工智能
xiaoyan20151 小时前
2026爆肝!Flutter3.41纯手撸微信聊天APP原生应用
android·flutter·dart