您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關Retrofit2.0怎么實現圖文上傳,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
最近項目里用到了類似圖文上傳的功能,以前都是封裝OkHttp的文件上傳功能,這次想換個姿勢,想用Retrofit2.0實現這樣的功能,本來以為挺簡單的,沒想到進入了深坑,連續調整了好幾種姿勢都報了同一個錯,接著網上類似的文章找了一大推,講得都是模棱兩可,或者對多參數格式不夠友好,最后還是去看了相關的源碼,自己把這個問題提出來解決了,在這里記錄一下。
一、定義網絡請求接口
public interface GoodsReturnApiService { @Multipart @POST(Compares.GOODS_RETURN_POST) //這里是自己post文件的地址 Observable<GoodsReturnPostEntity> postGoodsReturnPostEntitys(@PartMap Map<String, RequestBody> map, @Part List<MultipartBody.Part> parts); }
上面定義了一個接口用于上傳文件請求,有幾個注解需要說明一下, @Multipart這是Retrofit專門用于文件上傳的注解,需要配合@POST一起使用。
方法postGoodsReturnPostEntitys(@PartMap Map<String, RequestBody> map, @Part List<MultipartBody.Part> parts)第一個參數使用注解@PartMap用于多參數的情況,如果是單個參數也可使用注解@Part。
在類型Map<String, RequestBody>中,Map第一個泛型String是服務器接收用于文件上傳參數字段的Key,第二個泛型RequestBody是OkHttp3包裝的上傳參數字段的Value,這也是圖文上傳成功的關鍵所在。在后面會具體說到。
第二個參數使用注解@Part用于文件上傳,多文件上傳使用集合類型List<MultipartBody.Part>,單文件可以使用類型MultipartBody.Part,具體的使用同樣后面講。
這里著重說明一下,postGoodsReturnPostEntitys(@PartMap Map<String, RequestBody> map, @Part List<MultipartBody.Part> parts)方法參數這樣寫純屬個人習慣,你也可以直接使用一個參數postGoodsReturnPostEntitys(@PartMap Map<String, RequestBody> map),不過后面對RequestBody的處理方式也要跟著變化,這里就不詳細說了,只會介紹上面這種簡便清晰的方式。
二、初始化Retrofit
public class HttpRequestClient { public static final String TAG = "HttpRequestClientTAG"; private static Retrofit retrofit; private static OkHttpClient getOkHttpClient() { //日志顯示級別 HttpLoggingInterceptor.Level level= HttpLoggingInterceptor.Level.BODY; //新建log攔截器 HttpLoggingInterceptor loggingInterceptor=new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() { @Override public void log(String message) { Log.d(TAG, message); } }); loggingInterceptor.setLevel(level); //定制OkHttp OkHttpClient.Builder httpClientBuilder = new OkHttpClient .Builder(); //OkHttp進行添加攔截器loggingInterceptor httpClientBuilder.addInterceptor(loggingInterceptor); return httpClientBuilder.build(); } public static Retrofit getRetrofitHttpClient(){ if(null == retrofit){ synchronized (HttpRequestClient.class){ if(null == retrofit){ retrofit = new Retrofit.Builder() .client(getOkHttpClient()) .baseUrl(Compares.URL) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJava2CallAdapterFactory.create()) .build(); } } } return retrofit; } }
為了演示,Retrofit封裝比較簡陋,為的是查看網絡攔截,就不詳細說了。
三、發起文件上傳請求
private void postGoodsPicToServer(){ Map<String,RequestBody> params = new HashMap<>(); //以下參數是偽代碼,參數需要換成自己服務器支持的 params.put("type", convertToRequestBody("type")); params.put("title",convertToRequestBody("title")); params.put("info",convertToRequestBody("info"); params.put("count",convertToRequestBody("count")); //為了構建數據,同樣是偽代碼 String path2 = Environment.getExternalStorageDirectory() + File.separator + "test1.jpg"; String path3 = Environment.getExternalStorageDirectory() + File.separator + "test1.jpg"; List<File> fileList = new ArrayList<>(); fileList.add(new File(path2)); fileList.add(new File(path3)); List<MultipartBody.Part> partList = filesToMultipartBodyParts(fileList); HttpRequestClient.getRetrofitHttpClient().create(GoodsReturnApiService.class) .postGoodsReturnPostEntitys(params,partList) .subscribeOn(Schedulers.newThread()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(new Observer<GoodsReturnPostEntity>() { @Override public void onSubscribe(@NonNull Disposable d) { } @Override public void onNext(@NonNull GoodsReturnPostEntity goodsReturnPostEntity) { } @Override public void onError(@NonNull Throwable e) { } @Override public void onComplete() { } }); }
上面的params和fileList都是構造的偽代碼,需要根據自己項目的業務需求改變。
下面是上傳文件成功第一個關鍵,對參數請求頭(姑且叫這個名字,對應Retrofit上傳文件時參數那部分請求頭,下文件(圖片)請求頭同理,對應文件那部分請求頭)的content-type賦值,使用convertToRequestBody()方法。
private RequestBody convertToRequestBody(String param){ RequestBody requestBody = RequestBody.create(MediaType.parse("text/plain"), param); return requestBody; }
因為GsonConverterFactory.create()轉換器的緣故,會將參數請求頭的content-type值默認賦值application/json,如果沒有進行這步轉換操作,就可以在OKHttp3的日志攔截器中查看到這樣的賦值,這樣導致服務器不能正確識別參數,導致上傳失敗,所以這里需要對參數請求頭的content-type設置一個正確的值:text/plain。
下面是上傳文件成功第二個關鍵的地方,將文件(圖片)請求頭的content-type使用方法filesToMultipartBodyParts()對其賦值"image/png",并返回MultipartBody.Part集合。
private List<MultipartBody.Part> filesToMultipartBodyParts(List<File> files) { List<MultipartBody.Part> parts = new ArrayList<>(files.size()); for (File file : files) { RequestBody requestBody = RequestBody.create(MediaType.parse("image/png"), file); MultipartBody.Part part = MultipartBody.Part.createFormData("multipartFiles", file.getName(), requestBody); parts.add(part); } return parts; }
關于“Retrofit2.0怎么實現圖文上傳”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。