【问题标题】:Is there a way to add query parameter to every request with Retrofit 2?有没有办法使用 Retrofit 2 向每个请求添加查询参数?
【发布时间】:2015-10-05 11:58:32
【问题描述】:

我需要为 Retrofit 2.0.0-beta2 库发出的每个请求添加一个查询参数。我为 Retrofit 1.9 找到了this solution,但是如何在最新的库版本中添加RequestInterceptor

我的界面:

@GET("user/{id}")
Call<User> getUser(@Path("id")long id);

@GET("users/")
Call<List<User>> getUser();

客户:

Retrofit retrofit = new Retrofit.Builder()
                    .baseUrl(BASE_URL)
                    .addConverterFactory(GsonConverterFactory.create())
                    .client(CLIENT)  // custom OkHTTP Client
                    .build();
service = retrofit.create(userService.class);

【问题讨论】:

  • 如果您设法在拦截器中添加了查询参数,您可以在您的问题中添加代码还是在您自己的问题的响应中添加代码?

标签: java android retrofit


【解决方案1】:

为了完整起见,这里是您需要使用 OkHttp-Interceptor 为每个 Retrofit 2.x 请求添加参数的完整代码:

OkHttpClient client = new OkHttpClient();

client.interceptors().add(new Interceptor() {
    @Override
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();
        HttpUrl url = request.url().newBuilder().addQueryParameter("name","value").build();
        request = request.newBuilder().url(url).build();
        return chain.proceed(request);
    }
});

Retrofit retrofit = new Retrofit.Builder()
        .baseUrl("...")
        .client(client)
        .build();

【讨论】:

  • Request 对象中的httpUrl() 方法在最新版本的OkHttp 中更改为url()
  • interceptors()3.2.0 中返回不可变的List。在OkHttpClient.Builder 中使用addInterceptor()
  • 添加的参数将在GET中设置,如果您需要使用其他请求类型可能会出现问题。
【解决方案2】:

现在改造已发布 2.0.0,这是我的解决方案:

OkHttpClient client = new OkHttpClient.Builder()
        .addInterceptor(new Interceptor() {
            @Override
            public Response intercept(Chain chain) throws IOException {

                String uid = "0";
                long timestamp = (int) (Calendar.getInstance().getTimeInMillis() / 1000);
                String signature = MD5Util.crypt(timestamp + "" + uid + MD5_SIGN);
                String base64encode = signature + ":" + timestamp + ":" + uid;
                base64encode = Base64.encodeToString(base64encode.getBytes(), Base64.NO_WRAP | Base64.URL_SAFE);

                Request request = chain.request();
                HttpUrl url = request.url()
                        .newBuilder()
                        .addQueryParameter("pageSize", "2")
                        .addQueryParameter("method", "getAliasList")
                        .build();

                request = request
                        .newBuilder()
                        .addHeader("Authorization", "zui " + base64encode)
                        .addHeader("from_client", "ZuiDeer")
                        .url(url)
                        .build();

                Response response = chain.proceed(request);
                return response;
            }
        }).build();


Retrofit retrofit = new Retrofit.Builder()
        .baseUrl(ApiConstants.API_BASE_URL)
        .client(client)
        .addConverterFactory(GsonConverterFactory.create())
        .addCallAdapterFactory(RxJavaCallAdapterFactory.create())
        .build();

mRestfulService = retrofit.create(RestfulService.class);

【讨论】:

    【解决方案3】:

    3.2.0 及更高版本中,您应该在OkHttpClient.Builder 中使用addInterceptor()

    例如,Retrolambda:

    HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor()
            .setLevel(HttpLoggingInterceptor.Level.BASIC);
    
    Interceptor clientInterceptor = chain -> {
        Request request = chain.request();
        HttpUrl url = request.url().newBuilder().addQueryParameter("name", "value").build();
        request = request.newBuilder().url(url).build();
        return chain.proceed(request);
    };
    
    OkHttpClient client = new OkHttpClient.Builder()
            .addNetworkInterceptor(clientInterceptor)
            .addInterceptor(loggingInterceptor)
            .build();
    
    Retrofit retrofit = new Retrofit.Builder()
            .baseUrl(BASE_URL)
            .addConverterFactory(GsonConverterFactory.create())
            .client(client)
            .build();
    

    【讨论】:

      【解决方案4】:

      您必须从OkHttp 切换到Interceptor。创建一个OkHttpClient,将Interceptor 添加到它,并在Retrofit Builder 中传递该客户端。

      OkHttpClient client = new OkHttpClient();
      client.interceptors().add(new Interceptor() {
          @Override
          public Response intercept(Chain chain) throws IOException {
              ...
          }
      });
      
      Retrofit retrofit = new Retrofit.Builder()
              .baseUrl("...")
              .client(client)
              .build();
      

      然后,您可以使用 chain.request().newBuilder() 根据您的需要调整请求。详情请见documentation

      【讨论】:

      • 它绝对没有解释如何添加查询参数。只是解释了如何拦截,但我只找到了一种添加headers的方法,而OP和我想添加一个查询参数
      • 您可以根据自己的需要修改请求链,详情参见 OkHttp 文档。 (例如:chain.proceed(chain.request().newBuilder().url(chain.request().httpUrl().newBuilder().addQueryParameter("name", "value").build()).build ()); // square.github.io/okhttp/javadoc/com/squareup/okhttp/…)
      • 你是对的。随着 API 的变化,很难找到如何添加查询参数,所以你的解决方案对我没有帮助,直到我发现 httpurl 也有一个 newBuilder 方法。我认为您应该将其添加到您的问题中
      【解决方案5】:

      在 kotlin 中,将以下拦截器添加到您在 Retrofit builder 中设置的 OkHttpClient :

      Retrofit.Builder()
          .baseUrl("...")
          .client(
              OkHttpClient.Builder()
                  .addInterceptor { chain ->
                      val url = chain
                          .request()
                          .url()
                          .newBuilder()
                          .addQueryParameter("key", "value")
                          .build()
                      chain.proceed(chain.request().newBuilder().url(url).build())
                  }
                  .build()
          )
          .build()
          .create(FooService::class.java)
      

      当然,将 OkHttpClient 构建提取到 val 或注入依赖项中会使其更加模块化和易读。

      【讨论】:

        【解决方案6】:

        很多这些答案都是相似的,但我遇到的一个问题是 Interceptor 内的函数链接导致它失败。根据链接的视频更改cannot be made directly to a url。相反,必须制作 url 的副本,然后重新分配回原始 url,如下所示:

        {

        public method(){
            final String api_key = "key";
        
            OkHttpClient client = new OkHttpClient.Builder()
                                  .addInterceptor(new Interceptor() {
                                      @Override
                                      public Response intercept(Chain chain) throws IOException {
                                            Request original = chain.request();
                                            HttpUrl httpUrl = original.url();
        
                                            HttpUrl newHttpUrl = httpUrl
                                                                .newBuilder()
                                                                .addQueryParameter("api_key", api_key)
                                                                .build();
        
                                            Request.Builder requestBuilder = original
                                                                         .newBuilder()
                                                                     .url(newHttpUrl);
        
                                            Request request = requestBuilder
                                                              .build();
                                            return chain.proceed(request);
                                      }
                                  }).build();
        
        
            retrofit = new Retrofit.Builder()
                    .baseUrl("https://base.url.ext/")
                    .client(client)
                    .addConverterFactory(GsonConverterFactory.create())
                    .build();
        }
        

        虽然调用的函数与第一个答案相同,但此答案对函数调用进行了分区。即,原始 url 和新 url 存储在单独的局部变量中。这可以防止覆盖原始 url,直到您希望 OkHttpClient 这样做。

        【讨论】:

          【解决方案7】:

          对于更简洁的代码方法,为拦截器链设置一个单独的类,如下所示:

              public class LanguageInterceptor implements Interceptor {
                  @Override
                  public Response intercept(Chain chain) throws IOException {
                      HttpUrl url = chain.request()
                              .url()
                              .newBuilder()
                              .addQueryParameter("name","value")
                              .build();
          
                      Request request = chain.request()
                              .newBuilder()
                              .url(url)
                              .build();
                      Response response = chain.proceed(request);
                      return response;
                  }
              }
          

          然后在定义改造实例的类中,使用.addInterceptor(new LanguageInterceptor()) 向其中添加实例对象,如下所示:

          public static Retrofit getClient() {
                  HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor();
                  interceptor.setLevel(HttpLoggingInterceptor.Level.BODY);
                  OkHttpClient client = new OkHttpClient.Builder()
                          .connectTimeout(30, TimeUnit.SECONDS)
                          .readTimeout(30, TimeUnit.SECONDS)
                          .addInterceptor(new LanguageInterceptor())
                          .addInterceptor(interceptor)
                          .build();
          
                  if (retrofit == null) {
                      retrofit = new Retrofit.Builder()
                              .baseUrl(BASE_URL)
                              .client(client)
                              .addConverterFactory(ScalarsConverterFactory.create())
                              .addConverterFactory(GsonConverterFactory.create())
                              .build();
                      Log.e(TAG, "getClient: base url " + retrofit.baseUrl());
                  }
                  return retrofit;
              }
          

          【讨论】:

          • 太棒了!正是我需要的,我必须支持多种语言(在一个 biggg 项目中),这是最好的解决方案 :)
          猜你喜欢
          • 2015-12-12
          • 2016-09-03
          • 2018-03-29
          • 2017-06-27
          • 1970-01-01
          • 2020-01-01
          • 1970-01-01
          • 2013-07-08
          • 1970-01-01
          相关资源
          最近更新 更多