1.寫在前面的話
首先說明,我還沒太搞懂retrofit,目前,這篇博客只能給出這幾個內容。
- 文件上傳
- 文件下載
- 文件下載的進度監聽
還有這兩點沒弄好,
- 多文件一次上傳 (批量上傳)
- 文件上傳進度監聽
當前使用版本
compile 'com.squareup.retrofit2:retrofit:2.0.2'
2. 文件上傳
2.1 api 接口編寫
public interface uploadfileApi {
@Multipart
@POST("/fileabout.php")
Call<String> upload(@Part("fileName") String des,
@Part("file\"; filename=\"1.txt") RequestBody file);
}
- @Part("fileDes") String des 可以加一些描述信息(可以不加)
- @Part("file"; filename="1.txt") 格式不變,只需將1.text 對應的替換為你想在服務器生成的文件名稱
- 如果想傳多個文件,多次請求,當然,也可以像表單一樣(還沒弄好)
當然,上面這種辦法的靈活性差了點,我們可以選擇下面這種寫法
public interface uploadfileApi {
@Multipart
@POST("/fileabout.php")
Call<String> upload_2(@PartMap Map<String,RequestBody> params);
}
2.2 上傳文件
第一種api接口對應的代碼
Retrofit retrofit= new Retrofit.Builder()
.addConverterFactory(GsonConverterFactory.create())
.baseUrl("http://192.168.56.1")
.build();
uploadfileApi service =retrofit.create(uploadfileApi.class);
File file = new File(Environment.getExternalStorageDirectory() + "/" + "1.txt");
RequestBody requestBody = RequestBody.create(MediaType.parse("multipart/form-data"),file);
Call<String> model = service.upload("this is txt",requestBody);
model.enqueue(new Callback<String>() {
@Override
public void onResponse(Call<String> call, Response<String> response) {
Log.e(TAG, "onResponse: " + response.body().toString() );
}
@Override
public void onFailure(Call<String> call, Throwable t) {
}
});
- baseurl 為你的服務器地址,(我這里在局域網)
- file 文件為你手機中某個存在的文件
第二中API,我們只需要將相應第一種中的參數用map存起來,不多說了。
2.3 服務器接受文件
服務器接受文件的代碼就簡單多了,我這里以php為例
<?php
//var_dump($_POST);
//var_dump($_FILES);
$myfile = fopen("testfile.txt", "w");
fwrite($myfile, $_FILES["file"]["tmp_name"]."\n"
."D:\WWW"."\".$_FILES["file"]["name"]);
move_uploaded_file($_FILES["file"]["tmp_name"], "D:\WWW"."\".$_FILES["file"]["name"]);
> 上面這個代碼就是將文件的文件名寫入到textfile.txt文件中,并且將文件寫在當前d:\www\目錄下,文件名就是上傳的文件名。
結果如下如:

### 3. 文件下載及速度監聽
Retrofit并沒有給我們提供文件下載進度的相關信息,但是,我們還是可以從一些渠道知道如何監聽下載進度,在OKHTTP的官方demo里面有一個Progress.java的文件,從名字上就知道與進度有關。[github地址](https://github.com/square/okhttp/blob/master/samples/guide/src/main/java/okhttp3/recipes/Progress.java)
#### 3.1 改造改造ResponseBody
okhttp3默認的responsebody是不能滿足我們的要求的,(不能知道進度的相關信息),我們需要作出改造,首先需要個接口,監聽進度信息。其次,好吧,我承認這是廢話,我們只需要把Progress.java中我們需要的拿出來就好。
##### 3.1.1 interface
public interface ProgressListener {
/**
* @param progress 已經下載或上傳字節數
* @param total 總字節數
* @param done 是否完成
*/
void onProgress(long progress, long total, boolean done);
}
##### 3.1.2 ProgressResponseBody
public class ProgressResponseBody extends ResponseBody {
private final ResponseBody responseBody;
private final ProgressListener listener;
private BufferedSource bufferedSource;
public ProgressResponseBody(ResponseBody responseBody,ProgressListener listener){
this.responseBody = responseBody;
this.listener = listener;
}
@Override
public MediaType contentType() {
return responseBody.contentType();
}
@Override
public long contentLength() {
return responseBody.contentLength();
}
@Override
public BufferedSource source() {
if (null == bufferedSource){
bufferedSource = Okio.buffer(source(responseBody.source()));
}
return bufferedSource;
}
private Source source(Source source) {
return new ForwardingSource(source) {
long totalBytesRead = 0L;
@Override
public long read(Buffer sink, long byteCount) throws IOException {
long bytesRead = super.read(sink, byteCount);
totalBytesRead += bytesRead != -1 ? bytesRead : 0;
listener.onProgress(totalBytesRead, responseBody.contentLength(), bytesRead == -1);
return bytesRead;
}
};
}
}
恩,就是這些東西,別為我okio的相關知識,我也正在學呢。這個文件就是ophttp3的官方demo里面的東西。
#### 3.2 使用自己的okhttpclient
我們需要通過OkHttpClient的攔截器去攔截Response,并將我們的ProgressReponseBody設置進去,這樣才能監聽進度。那么,我們怎么講client設置進去呢。通過觀察Retrofit的結構發現,Builder下面有client()方法可以設置,好,那么我們通過Retrofit.Builder來創建(這樣我們可以配置了)。

相關代碼如下
Retrofit.Builder builder = new Retrofit.Builder()
.baseUrl("http://192.168.56.1");
OkHttpClient client = new OkHttpClient.Builder()
.addNetworkInterceptor(new Interceptor() {
@Override
public okhttp3.Response intercept(Chain chain) throws IOException {
okhttp3.Response orginalResponse = chain.proceed(chain.request());
return orginalResponse.newBuilder()
.body(new ProgressResponseBody(orginalResponse.body(), new ProgressListener() {
@Override
public void onProgress(long progress, long total, boolean done) {
Log.e(TAG, Looper.myLooper()+"");
Log.e(TAG, "onProgress: " + "total ---->" + total + "done ---->" + progress );
}
}))
.build();
}
})
.build();
DownLoadApi api = builder.client(client)
.build().create(DownLoadApi.class);
** 注意進度的監聽發生在子線程中,要切記**
#### 3.3 將response寫入到文件里
寫入的操作就簡單了,代碼如下,沒什么好說的。
Call<ResponseBody> call = api.getFile("image_text.png");
call.enqueue(new Callback<ResponseBody>() {
@Override
public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
try {
InputStream is = response.body().byteStream();
File file = new File(Environment.getExternalStorageDirectory(), "text_img.png");
FileOutputStream fos = new FileOutputStream(file);
BufferedInputStream bis = new BufferedInputStream(is);
byte[] buffer = new byte[1024];
int len;
while ((len = bis.read(buffer)) != -1) {
fos.write(buffer, 0, len);
fos.flush();
}
fos.close();
bis.close();
is.close();
} catch (IOException e) {
e.printStackTrace();
}
Log.e(TAG,"success");
}
@Override
public void onFailure(Call<ResponseBody> call, Throwable t) {
}
});
** 注意image_text.png是我事先將這張圖片放入到相應路徑下面的,如圖,要確定能訪問到才行 **

#### 3.4 最后結果展示

### 4. 總結
retrofit的功能強大,靈活性強,但是這就意味著使用起來稍微麻煩一點(至少我是這樣認為的),但是,retrofit依賴于okhttp,okhttp是有demo供我們學習的,so,學習demo去吧,[鏈接地址](https://github.com/square/okhttp/tree/master/samples)