Android 中 OkHttp 的自定义 Interceptor 实现统一请求头添加

在 Android 中使用 OkHttp 的自定义 Interceptor 实现统一请求头添加是一种高效的方式,可以避免在每个请求中重复设置头部。以下是详细实现步骤:

1. 创建自定义拦截器类

java 复制代码
import okhttp3.Interceptor;
import okhttp3.Request;
import okhttp3.Response;
import java.io.IOException;

public class HeaderInterceptor implements Interceptor {
    @Override
    public Response intercept(Chain chain) throws IOException {
        // 1. 获取原始请求
        Request originalRequest = chain.request();
        
        // 2. 添加统一请求头(可扩展更多)
        Request newRequest = originalRequest.newBuilder()
                .header("User-Agent", "MyApp/Android/1.0.0")
                .header("Accept-Language", "zh-CN")
                .header("Authorization", getAuthToken()) // 示例:动态token
                .header("Custom-Header", "Value")
                // 添加更多统一头部...
                .build();
        
        // 3. 继续执行请求
        return chain.proceed(newRequest);
    }

    // 示例方法:获取动态认证token(根据实际情况实现)
    private String getAuthToken() {
        // 这里可以是获取本地存储的token
        return "Bearer your_token_here";
    }
}

2. 在 OkHttpClient 中注册拦截器

java 复制代码
// 创建 OkHttpClient 实例
OkHttpClient client = new OkHttpClient.Builder()
        .addInterceptor(new HeaderInterceptor()) // 添加统一头部拦截器
        // 可选:添加其他拦截器(如日志拦截器)
        .addInterceptor(new HttpLoggingInterceptor().setLevel(HttpLoggingInterceptor.Level.BODY))
        .build();

3. 使用 OkHttpClient 发送请求(示例)

java 复制代码
// 创建请求(无需单独设置统一头部)
Request request = new Request.Builder()
        .url("https://api.example.com/data")
        .build();

// 异步执行
client.newCall(request).enqueue(new Callback() {
    @Override
    public void onFailure(Call call, IOException e) {
        // 处理失败
    }

    @Override
    public void onResponse(Call call, Response response) throws IOException {
        // 处理响应
    }
});

关键点说明

  1. 拦截器类型选择

    • addInterceptor:应用拦截器,处理非重定向请求
    • addNetworkInterceptor:网络拦截器,处理实际网络请求(包含重定向)
  2. 动态头部处理

    java 复制代码
    // 示例:根据请求URL动态添加头部
    if (originalRequest.url().toString().contains("/api/secure")) {
        newRequestBuilder.header("Security-Key", "special_key");
    }
  3. 覆盖优先级

    • 拦截器添加的头部 会覆盖 请求中设置的相同名称头部

    • 若需保留请求特定头部,使用以下方式:

      java 复制代码
      if (!originalRequest.headers().contains("Special-Header")) {
          newRequestBuilder.header("Special-Header", "default");
      }

高级场景处理

1. 需要动态 token 的场景
java 复制代码
.header("Authorization", getCurrentToken()) // 实现动态获取

// 实现示例
private String getCurrentToken() {
    return App.getAuthManager().getAccessToken();
}
2. 条件性添加头部
java 复制代码
Request.Builder builder = originalRequest.newBuilder();
if (isUserLoggedIn()) {
    builder.header("User-ID", getUserId());
}
3. 多模块差异化处理
java 复制代码
String path = originalRequest.url().encodedPath();
if (path.startsWith("/api/v1")) {
    builder.header("API-Version", "v1");
} else if (path.startsWith("/api/v2")) {
    builder.header("API-Version", "v2");
}

完整流程图

是 否 发起请求 拦截器处理 是否添加统一头部? 添加预设头部 保留原始头部 执行网络请求 返回响应

通过这种方式,您可以:

  1. 集中管理所有公共请求头
  2. 动态处理认证信息
  3. 根据不同请求条件添加不同头部
  4. 减少代码重复和维护成本

提示:对于敏感信息(如认证 token),建议结合 Android 的安全存储机制(如 EncryptedSharedPreferences)进行存储和获取。

相关推荐
贫民窟的勇敢爷们15 小时前
SpringBoot整合AOP切面编程实战,实现日志统一记录+接口权限校验
java·spring boot·spring
AC赳赳老秦16 小时前
供应链专员提效:OpenClaw自动跟踪物流信息、更新库存数据,异常自动提醒
java·大数据·服务器·数据库·人工智能·自动化·openclaw
迈巴赫车主16 小时前
Java基础:list、set、map一遍过
java·开发语言
程序员陆业聪16 小时前
两次Flutter全屏白踩坑复盘:Layout的静默失败,以及AI结对编程的认知盲区
android
灵犀学长16 小时前
基于 Spring ThreadPoolTaskScheduler + CronTrigger 实现的动态定时任务调度系统
java·数据库·spring
程序员陆业聪17 小时前
Compose Strong Skipping Mode 的真相:它并不会让你的类型变 Stable
android
好家伙VCC18 小时前
【无标题】
java
小碗羊肉19 小时前
【JavaWeb | 第十一篇】文件上传(本地&阿里云OSS)
java·阿里云·servlet
吾疾唯君医19 小时前
Java SpringBoot集成积木报表实操记录
java·spring boot·spring·导出excel·积木报表·数据文件下载
Byron Loong19 小时前
【c++】为什么有了dll和.h,还需要包含lib
java·开发语言·c++