1. 程式人生 > >OKHttp使用詳解,步驟挺詳細的,適合初學者使用!

OKHttp使用詳解,步驟挺詳細的,適合初學者使用!

一,OKHttp介紹

okhttp是一個第三方類庫,用於android中請求網路。

這是一個開源專案,是安卓端最火熱的輕量級框架,由移動支付Square公司貢獻(該公司還貢獻了Picasso和LeakCanary) 。用於替代HttpUrlConnection和Apache HttpClient(android API23 裡已移除HttpClient)。

okhttp有自己的官網,官網網址:OKHttp官網

在AndroidStudio中使用不需要下載jar包,直接新增依賴即可: 
compile ‘com.squareup.okhttp3:okhttp:3.4.1’

下面對以OKHttp3來詳細介紹OKHttp的使用方法。

二,get請求的使用方法

使用OKHttp進行網路請求支援兩種方式,一種是同步請求,一種是非同步請求。下面分情況進行介紹。

1,get的同步請求

對於同步請求在請求時需要開啟子執行緒,請求成功後需要跳轉到UI執行緒修改UI。 
使用示例如下:

public void getDatasync(){
    new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件
Request request = new Request.Builder() .url("http://www.baidu.com")//請求介面。如果需要傳參拼接到介面後面。 .build();//建立Request 物件 Response response = null; response = client.newCall(request).execute();//得到Response 物件 if
(response.isSuccessful()) { Log.d("kwwl","response.code()=="+response.code()); Log.d("kwwl","response.message()=="+response.message()); Log.d("kwwl","res=="+response.body().string()); //此時的程式碼執行在子執行緒,修改UI的操作請使用handler跳轉到UI執行緒。 } } catch (Exception e) { e.printStackTrace(); } } }).start(); }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24

此時列印結果如下: 
response.code()==200; 
response.message()==OK; 
res=={“code”:200,”message”:success};

注意事項: 
1,Response.code是http響應行中的code,如果訪問成功則返回200.這個不是伺服器設定的,而是http協議中自帶的。res中的code才是伺服器設定的。注意二者的區別。 
2,response.body().string()本質是輸入流的讀操作,所以它還是網路請求的一部分,所以這行程式碼必須放在子執行緒。 
3,response.body().string()只能呼叫一次,在第一次時有返回值,第二次再呼叫時將會返回null。原因是:response.body().string()的本質是輸入流的讀操作,必須有伺服器的輸出流的寫操作時客戶端的讀操作才能得到資料。而伺服器的寫操作只執行一次,所以客戶端的讀操作也只能執行一次,第二次將返回null。

2,get的非同步請求

這種方式不用再次開啟子執行緒,但回撥方法是執行在子執行緒中,所以在更新UI時還要跳轉到UI執行緒中。 
使用示例如下:

private void getDataAsync() {
    OkHttpClient client = new OkHttpClient();
    Request request = new Request.Builder()
            .url("http://www.baidu.com")
            .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 {
            if(response.isSuccessful()){//回撥的方法執行在子執行緒。
                Log.d("kwwl","獲取資料成功了");
                Log.d("kwwl","response.code()=="+response.code());
                Log.d("kwwl","response.body().string()=="+response.body().string());
            }
        }
    });
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

非同步請求的列印結果與注意事項與同步請求時相同。最大的不同點就是非同步請求不需要開啟子執行緒,enqueue方法會自動將網路請求部分放入子執行緒中執行。

注意事項: 
1,回撥介面的onFailure方法和onResponse執行在子執行緒。 
2,response.body().string()方法也必須放在子執行緒中。當執行這行程式碼得到結果後,再跳轉到UI執行緒修改UI。

三,post請求的使用方法

Post請求也分同步和非同步兩種方式,同步與非同步的區別和get方法類似,所以此時只講解post非同步請求的使用方法。 
使用示例如下:

private void postDataWithParame() {
    OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件。
    FormBody.Builder formBody = new FormBody.Builder();//建立表單請求體
    formBody.add("username","zhangsan");//傳遞鍵值對引數
    Request request = new Request.Builder()//建立Request 物件。
            .url("http://www.baidu.com")
            .post(formBody.build())//傳遞請求體
            .build();
    client.newCall(request).enqueue(new Callback() {。。。});//回撥方法的使用與get非同步請求相同,此時略。
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

看完程式碼我們會發現:post請求中並沒有設定請求方式為POST,回憶在get請求中也沒有設定請求方式為GET,那麼是怎麼區分請求方式的呢?重點是Request.Builder類的post方法,在Request.Builder物件建立最初預設是get請求,所以在get請求中不需要設定請求方式,當呼叫post方法時把請求方式修改為POST。所以此時為POST請求。

四,POST請求傳遞引數的方法總結

在post請求使用方法中講了一種傳遞引數的方法,就是建立表單請求體物件,然後把表單請求體物件作為post方法的引數。post請求傳遞引數的方法還有很多種,但都是通過post方法傳遞的。下面我們看一下Request.Builder類的post方法的宣告:

public Builder post(RequestBody body)
  • 1

由方法的宣告可以看出,post方法接收的引數是RequestBody 物件,所以只要是RequestBody 類以及子類物件都可以當作引數進行傳遞。FormBody就是RequestBody 的一個子類物件。

1,使用FormBody傳遞鍵值對引數

這種方式用來上傳String型別的鍵值對 
使用示例如下:

private void postDataWithParame() {
    OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件。
    FormBody.Builder formBody = new FormBody.Builder();//建立表單請求體
    formBody.add("username","zhangsan");//傳遞鍵值對引數
    Request request = new Request.Builder()//建立Request 物件。
            .url("http://www.baidu.com")
            .post(formBody.build())//傳遞請求體
            .build();
    client.newCall(request).enqueue(new Callback() {。。。});//此處省略回撥方法。
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

2,使用RequestBody傳遞Json或File物件

RequestBody是抽象類,故不能直接使用,但是他有靜態方法create,使用這個方法可以得到RequestBody物件。

這種方式可以上傳Json物件或File物件。 
上傳json物件使用示例如下:

OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件。
MediaType JSON = MediaType.parse("application/json; charset=utf-8");//資料型別為json格式,
String jsonStr = "{\"username\":\"lisi\",\"nickname\":\"李四\"}";//json資料.
RequestBody body = RequestBody.create(JSON, josnStr);
Request request = new Request.Builder()
        .url("http://www.baidu.com")
        .post(body)
        .build();
client.newCall(request).enqueue(new Callback() {。。。});//此處省略回撥方法。
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

上傳File物件使用示例如下:

OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件。
MediaType fileType = MediaType.parse("File/*");//資料型別為json格式,
File file = new File("path");//file物件.
RequestBody body = RequestBody.create(fileType , file );
Request request = new Request.Builder()
        .url("http://www.baidu.com")
        .post(body)
        .build();
client.newCall(request).enqueue(new Callback() {。。。});//此處省略回撥方法。
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

3,使用MultipartBody同時傳遞鍵值對引數和File物件

這個字面意思是多重的body。我們知道FromBody傳遞的是字串型的鍵值對,RequestBody傳遞的是多媒體,那麼如果我們想二者都傳遞怎麼辦?此時就需要使用MultipartBody類。 
使用示例如下:

OkHttpClient client = new OkHttpClient();
MultipartBody multipartBody =new MultipartBody.Builder()
        .setType(MultipartBody.FORM)
        .addFormDataPart("groupId",""+groupId)//新增鍵值對引數
        .addFormDataPart("title","title")
        .addFormDataPart("file",file.getName(),RequestBody.create(MediaType.parse("file/*"), file))//新增檔案
        .build();
final Request request = new Request.Builder()
        .url(URLContant.CHAT_ROOM_SUBJECT_IMAGE)
        .post(multipartBody)
        .build();
client.newCall(request).enqueue(new Callback() {。。。});
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

4,自定義RequestBody實現流的上傳

在上面的分析中我們知道,只要是RequestBody類以及子類都可以作為post方法的引數,下面我們就自定義一個類,繼承RequestBody,實現流的上傳。 
使用示例如下: 
首先建立一個RequestBody類的子類物件:

RequestBody body = new RequestBody() {
    @Override
    public MediaType contentType() {
        return null;
    }

    @Override
    public void writeTo(BufferedSink sink) throws IOException {//重寫writeTo方法
        FileInputStream fio= new FileInputStream(new File("fileName"));
        byte[] buffer = new byte[1024*8];
        if(fio.read(buffer) != -1){
             sink.write(buffer);
        }
    }
};
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

然後使用body物件:

 OkHttpClient client = new OkHttpClient();//建立OkHttpClient物件。
Request request = new Request.Builder()
        .url("http://www.baidu.com")
        .post(body)
        .build();
client.newCall(request).enqueue(new Callback() {。。。});
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

以上程式碼的與眾不同就是body物件,這個body物件重寫了write方法,裡面有個sink物件。這個是OKio包中的輸出流,有write方法。使用這個方法我們可以實現上傳流的功能。

使用RequestBody上傳檔案時,並沒有實現斷點續傳的功能。我可以使用這種方法結合RandomAccessFile類實現斷點續傳的功能。

五,設定請求頭

OKHttp中設定請求頭特別簡單,在建立request物件時呼叫一個方法即可。 
使用示例如下:

Request request = new Request.Builder()
                .url("http://www.baidu.com")
                .header("User-Agent", "OkHttp Headers.java")
                .addHeader("token", "myToken")
                .build();
  • 1
  • 2
  • 3
  • 4
  • 5

其他部分程式碼略。

六,下載檔案

在OKHttp中並沒有提供下載檔案的功能,但是在Response中可以獲取流物件,有了流物件我們就可以自己實現檔案的下載。程式碼如下: 
這段程式碼寫在回撥介面CallBack的onResponse方法中:

try{
    InputStream  is = response.body().byteStream();//從伺服器得到輸入流物件
    long sum = 0;
    File dir = new File(mDestFileDir);
    if (!dir.exists()){
        dir.mkdirs();
    }
    File file = new File(dir, mdestFileName);//根據目錄和檔名得到file物件
    FileOutputStream  fos = new FileOutputStream(file);
    byte[] buf = new byte[1024*8];
    int len = 0;
    while ((len = is.read(buf)) != -1){
        fos.write(buf, 0, len);
    }
    fos.flush();
    return file;

}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

七,對於OKHttp的使用封裝

由於okhttp是偏底層的網路請求類庫,返回結果的回撥方法仍然執行在子執行緒中,需要自己跳轉到UI執行緒,使用麻煩。為了使用方便需要對OKHttp進行再次封裝。對於OKHttp的封裝首推的就是hongyang大神的OKHttpUtils。我個人在看過OKHttp的原碼和借鑑各大神的封裝原始碼後封裝了一套自己的OKHttpUtils。這套OKHttpUtils最大的優點是簡單和便於使用,這是我專案中實際用的網路請求工具類,完全可以說拿來即用。而且程式碼簡單,可供學習使用。

封裝的功能有: 
* 一般的get請求 
* 一般的post請求 
* 上傳單個檔案(包含進度) 
* 上傳list集合檔案 
* 上傳map集合檔案 
* 檔案下載(包含進度) 
* 圖片下載(實現了圖片的壓縮)