1. 程式人生 > >Okhttp 請求新增公共引數、公共Headers 方法

Okhttp 請求新增公共引數、公共Headers 方法

安卓開發中,http 網路請求的框架有很多,有 HttpUrlConnectionVolleyOkhttp很多種,現在由於 retrofit 和 rxjava 的出現, Okhttp 的使用越來越多,因為 Retrofit 也僅僅是在 Okhttp 做的一層封裝,那麼需求上需要我們在每一個 URL 請求中新增一個固定的引數時候,或者將請求的訊息頭統一改為我們需要的形式,我們要怎麼實現呢?

我們用過 OkHttp 的都知道它獨特而且強大的攔截器功能,比如 提供的 HttpLoggingInterceptor(),我們也可以通過自定義 攔截器來實現上述的公共引數需求。首先我們實現 Interceptor

介面

public interface Interceptor {
  Response intercept(Chain chain) throws IOException;

  interface Chain {
    Request request();

    Response proceed(Request request) throws IOException;

    /**
     * Returns the connection the request will be executed on. This is only available in the chains
     * of network interceptors; for application interceptors this is always null.
     */
@Nullable Connection connection(); } }

intercept (Chain chain) 方法中 Chain 物件可以拿到當前請求的 Request 物件,然後我們可以對Request做二次處理,最後生成我們需要的請求,然後再通過網路傳送請求到服務端,這樣就完成了一次攔截。由於程式碼很簡單,我們先介紹使用方法,原始碼貼在了後面:

//新增post公共請求引數  Constants.PACKAGE 和 Constants.CFROM
BasicParamsInterceptor basicParamsInterceptor = new BasicParamsInterceptor.Builder
() .addParam("from", "android") //新增公共引數到 post 請求體 .addQueryParam("version","1") // 新增公共版本號,加在 URL 後面 .addHeaderLine("X-Ping: Pong") // 示例: 新增公共訊息頭 .addParamsMap(map) // 可以新增 Map 格式的引數 .build();

然後在 OkHttpClient中新增攔截器就完成了

okHttpClient = new OkHttpClient.Builder()
                    .readTimeout(READ_TIME_OUT, TimeUnit.MILLISECONDS)
                    .connectTimeout(CONNECT_TIME_OUT, TimeUnit.MILLISECONDS)
                    .writeTimeout(CONNECT_TIME_OUT, TimeUnit.SECONDS)
                    .addInterceptor(basicParamsInterceptor) // 新增公共引數攔截器
                    .addInterceptor(logInterceptor)
                    .addInterceptor(noNetInterceptor)
                    .addNetworkInterceptor(cacheInterceptor)
                    .retryOnConnectionFailure(true)
                    .cache(cache)
                    .build();

整個攔截器原始碼如下:

/**
 * Created by :   Liu
 * date:         2017/2/16 14:04   <br/>
 */
public class BasicParamsInterceptor implements Interceptor {

    Map<String, String> queryParamsMap = new HashMap<>(); // 新增到 URL 末尾,Get Post 方法都使用
    Map<String, String> paramsMap = new HashMap<>(); // 新增到公共引數到訊息體,適用 Post 請求
    Map<String, String> headerParamsMap = new HashMap<>(); // 公共 Headers 新增
    List<String> headerLinesList = new ArrayList<>(); // 訊息頭 集合形式,一次新增一行

    // 私有構造器
    private BasicParamsInterceptor() {}

    @Override
    public Response intercept(Chain chain) throws IOException {

        Request request = chain.request();
        Request.Builder requestBuilder = request.newBuilder();

        // process header params inject
        Headers.Builder headerBuilder = request.headers().newBuilder();
        // 以 Entry 新增訊息頭
        if (headerParamsMap.size() > 0) {
            Iterator iterator = headerParamsMap.entrySet().iterator();
            while (iterator.hasNext()) {
                Map.Entry entry = (Map.Entry) iterator.next();
                headerBuilder.add((String) entry.getKey(), (String) entry.getValue());
            }
            requestBuilder.headers(headerBuilder.build());
        }
        // 以 String 形式新增訊息頭
        if (headerLinesList.size() > 0) {
            for (String line: headerLinesList) {
                headerBuilder.add(line);
            }
            requestBuilder.headers(headerBuilder.build());
        }
        // process header params end


        // process queryParams inject whatever it's GET or POST
        if (queryParamsMap.size() > 0) {
            request = injectParamsIntoUrl(request.url().newBuilder(), requestBuilder, queryParamsMap);
        }

        // process post body inject
        if (paramsMap.size() > 0) {
            if (canInjectIntoBody(request)) {
                FormBody.Builder formBodyBuilder = new FormBody.Builder();
                for(Map.Entry<String, String> entry : paramsMap.entrySet()) {
                    formBodyBuilder.add((String) entry.getKey(), (String) entry.getValue());
                }

                RequestBody formBody = formBodyBuilder.build();
                String postBodyString = bodyToString(request.body());
                postBodyString += ((postBodyString.length() > 0) ? "&" : "") +  bodyToString(formBody);
                requestBuilder.post(RequestBody.create(MediaType.parse("application/x-www-form-urlencoded;charset=UTF-8"), postBodyString));
            }
        }
        request = requestBuilder.build();
        return chain.proceed(request);
    }

    /**
     *  確認是否是 post 請求
     * @param request 發出的請求
     * @return true 需要注入公共引數
     */
    private boolean canInjectIntoBody(Request request) {
        if (request == null) {
            return false;
        }
        if (!TextUtils.equals(request.method(), "POST")) {
            return false;
        }
        RequestBody body = request.body();
        if (body == null) {
            return false;
        }
        MediaType mediaType = body.contentType();
        if (mediaType == null) {
            return false;
        }
        if (!TextUtils.equals(mediaType.subtype(), "x-www-form-urlencoded")) {
            return false;
        }
        return true;
    }

    // func to inject params into url
    private Request injectParamsIntoUrl(HttpUrl.Builder httpUrlBuilder, Request.Builder requestBuilder, Map<String, String> paramsMap) {
        if (paramsMap.size() > 0) {
            Iterator iterator = paramsMap.entrySet().iterator();
            while (iterator.hasNext()) {
                Map.Entry entry = (Map.Entry) iterator.next();
                httpUrlBuilder.addQueryParameter((String) entry.getKey(), (String) entry.getValue());
            }
            requestBuilder.url(httpUrlBuilder.build());
            return requestBuilder.build();
        }
        return null;
    }

    private static String bodyToString(final RequestBody request){
        try {
            final RequestBody copy = request;
            final Buffer buffer = new Buffer();
            if(copy != null)
                copy.writeTo(buffer);
            else
                return "";
            return buffer.readUtf8();
        }
        catch (final IOException e) {
            return "did not work";
        }
    }

    public static class Builder {

        BasicParamsInterceptor interceptor;

        public Builder() {
            interceptor = new BasicParamsInterceptor();
        }

        // 新增公共引數到 post 訊息體
        public Builder addParam(String key, String value) {
            interceptor.paramsMap.put(key, value);
            return this;
        }

        // 新增公共引數到 post 訊息體
        public Builder addParamsMap(Map<String, String> paramsMap) {
            interceptor.paramsMap.putAll(paramsMap);
            return this;
        }

        // 新增公共引數到訊息頭
        public Builder addHeaderParam(String key, String value) {
            interceptor.headerParamsMap.put(key, value);
            return this;
        }

        // 新增公共引數到訊息頭
        public Builder addHeaderParamsMap(Map<String, String> headerParamsMap) {
            interceptor.headerParamsMap.putAll(headerParamsMap);
            return this;
        }

        // 新增公共引數到訊息頭
        public Builder addHeaderLine(String headerLine) {
            int index = headerLine.indexOf(":");
            if (index == -1) {
                throw new IllegalArgumentException("Unexpected header: " + headerLine);
            }
            interceptor.headerLinesList.add(headerLine);
            return this;
        }

        // 新增公共引數到訊息頭
        public Builder addHeaderLinesList(List<String> headerLinesList) {
            for (String headerLine: headerLinesList) {
                int index = headerLine.indexOf(":");
                if (index == -1) {
                    throw new IllegalArgumentException("Unexpected header: " + headerLine);
                }
                interceptor.headerLinesList.add(headerLine);
            }
            return this;
        }

        // 新增公共引數到 URL 
        public Builder addQueryParam(String key, String value) {
            interceptor.queryParamsMap.put(key, value);
            return this;
        }

        // 新增公共引數到 URL 
        public Builder addQueryParamsMap(Map<String, String> queryParamsMap) {
            interceptor.queryParamsMap.putAll(queryParamsMap);
            return this;
        }

        public BasicParamsInterceptor build() {
            return interceptor;
        }

    }
}