android 网络访问拦截器使用后的bug处理

客户端调用header鉴权

说明:登录成功以后,会把JSESSIONID保存到AppDataStore中,然后我们在去访问接口的时候,自动的由拦截器去帮我们增加header。

一般代码如下:

复制代码
override fun intercept(chain: Interceptor.Chain): Response {
        val request = chain.request()
        if (request.url.toString().indexOf("login") != -1) return chain.proceed(request)

        var jsessionId = cachedJSessionId
        if (jsessionId == null) {
            synchronized(updateLock) {
                jsessionId = cachedJSessionId
                if (jsessionId == null) {
                    updateJSessionId()
                    return chain.proceed(request)
                }
            }
        }

        val cookieHeader = request.header("Cookie").orEmpty()
        val newCookie = if (cookieHeader.isEmpty()) {
            "JSESSIONID=$jsessionId"
        } else {
            "$cookieHeader; JSESSIONID=$jsessionId"
        }

        val newRequest = request.newBuilder()
            .header("Cookie", newCookie)
            .build()

        return chain.proceed(newRequest)
    }

错误出现并查找原因

开始一切都是好好的,但是最近访问10次,只有2次是正常的,数据加载成功,其它的8次基本上是报405错误。

然后我们来看看具体的405的说明:

Android 访问接口报 405 是一个很常见的客户端错误,它的核心原因非常明确:客户端发送的 HTTP 请求方法(Method),服务器不允许或不支持。

核心原因:方法不匹配

HTTP 1.1 规范定义了多种请求方法,最常用的有:

  • GET: 从服务器获取资源。
  • POST: 向服务器提交数据,创建新资源。
  • PUT: 向服务器提交数据,更新已存在的资源。
  • DELETE: 请求服务器删除指定资源。
  • PATCH: 对资源进行部分更新。
  • HEAD: 只请求资源的头部信息,不返回具体内容。

当你在 Android 端使用了一个服务器接口不支持的方法时,服务器就会返回 405 Method Not Allowed 错误。

举个例子:

  • 服务器上的某个接口 /api/login 只接受 POST 请求(因为需要提交用户名和密码)。
  • 你的 Android 代码中却错误地使用了 GET 方法去访问它。
  • 服务器收到 GET /api/login 请求后,发现这个接口不支持 GET,于是就返回 405 错误。

解决方案步骤(由易到难)

遇到 405 错误,最直接的排查方向就是确认请求方法是否正确

1. 核对接口文档,确认正确的请求方法

这是最重要的一步。你必须与后端开发人员或接口文档确认,你要访问的这个接口究竟应该使用哪种 HTTP 方法。

2. 检查你的 Android 代码,确保方法正确

根据你使用的网络库,检查代码中的请求方法设置。

示例 1:使用 Retrofit (最常用)

Retrofit 通过注解来指定请求方法。

java

运行

复制代码
// 假设接口文档规定 /api/data 应该使用 POST 方法
public interface ApiService {
    // 正确的写法
    @POST("/api/data")
    Call<MyResponse> fetchData(@Body MyRequest request);

    // 错误的写法 (如果服务器不支持 GET,就会报 405)
    @GET("/api/data") 
    Call<MyResponse> fetchDataError(@Query("param") String param);
}

排查点: 仔细检查你的 ApiService 接口中对应方法的注解是 @GET@POST 还是其他,并与文档保持一致。

这里面最主要是405是header中的JSESSIONID不对,是旧的,还是上次登录的时候返回的,并且我们在新的登录成功以后,又将旧的JSESSIONID传入到了服务器去请求API导致。

问题处理

在拦截器中定义清空的方法,每次登录成功以后,去执行一次,保存我们的拦截器去读最新的。

复制代码
// 在 AuthInterceptor 类中添加此方法
    fun clearCachedJSessionId() {
        synchronized(updateLock) {
            cachedJSessionId = null
        }
        updateJSessionId() // 立即重新加载
    }

在登录成功以后,我们去调用一下:

说明:先将要用到authInterceptor的地方进行注入,然后我们再去调用即可。

这次每次成功的获取了新的JSESSIONID,都去处理一下,保证最新的header传入就可以。

总结

与服务器(shiro)、(spring security)交互的时候,我们都需要保存一些标识信息(auth),然后用这个auth信息(JSEESIONID)去与服务器进行交互,这样可以保证接口调用的正确性,也可以保存唯一性。android中我们使用拦截器的方式就可以达到自动处理请求header。

拦截器的几种方式:

https://blog.csdn.net/jwbabc/article/details/149103966?spm=1011.2415.3001.5331

有兴趣的小伙伴可以自己试一下。

相关推荐
王家视频教程图书馆4 小时前
修复服务端500相应,修复客户端上传文件.tmp 服务端接受不到文件bug
bug
qq_401700415 小时前
Qt开发过程中遇到哪些经典的bug
qt·bug
0白露2 天前
关闭搜狗输入法右下角广告,可以适用于大多数应用系统通知的广告
windows·bug
一只自律的鸡3 天前
【Linux驱动】bug处理 ens33找不到IP
linux·运维·bug
Lichenpar4 天前
Springboot采用FastJson2作为MessageConverter时,配置的全局日期类型序列化转换BUG
java·开发语言·bug
Irene19915 天前
AI 找 bug 的局限性 和 Deepseek 优势明显
bug
癫狂的兔子5 天前
【bug】【Python】pandas中的DataFrame.to_excel()和ExcelWriter的区别
python·bug
癫狂的兔子6 天前
【BUG】【Python】【Spider】Compound class names are not allowed.
开发语言·python·bug
netkiller-BG7NYT6 天前
阿里云语音合成CosyVoice重大BUG被我发现了
bug
马猴烧酒.7 天前
【Mybatis出现bug】应为 <statement> 或 DELIMITER,得到 ‘id‘
java·bug·mybatis