目录

 

前言

一、效果展示

二、基本配置

三、代码实战

3.1、创建RetrofitManager和APIService

3.2、准备好选择的图片

3.3、开始构造参数

3.4、实现上传

附:UploadHelper.java源码


前言

距离上一篇文章到现在已经有将近半年的时间了,因为换了一座城市,到现在才算是刚刚熟悉起来吧,所以这段时间一直没能静下心来去总结,今天是周末,首先祝大家都能度过一个开心快乐的周末时光,我这个屌丝只能坐在家里码码文字了。今天总结一下关于在Android开发中使用Retrofit2实现类似于Web端表单文件上传的技术实现。

工作了也有将近三年的时间了,这期间关于这一块的实现方法也是换了几波,从最开始使用的基于Volley开发的MultipartRequest,到后来的OkHttp3,再到现在使用的Retrofit2,总结下来,其实都是大同小异。客户端这边主要就是构造请求参数,不同的参数需要指定其对应的参数类型,具体体现就是在构造参数时,需要指定Content-Type的类型(如果你平时观察过http请求,肯定会在Request Headers就是请求头中看到过它,我下面的日志截图里也有),然后构造完参数以后,剩下的就是和发送普通的Http请求一样了。服务端会有一个叫做enctype的东西,它的作用是告知服务器请求正文的MIME类型(和请求消息头的headers作用一样),然后服务端就根据对应的类型去解析,比如普通文字类型直接根据对应实体中的字段获取对应的值,文件类型的通过IO流读取写入文件,这样就完成了整个的上传流程。那么接下来,就来具体说说Android客户端Retrofit2的实现过程,至于服务端因为我懂得不多,所以就不在这里丢人了,简单点的大家可以尝试着写个Servlet去测试测试。

一、效果展示

这里录了一个gif效果图,可以很清晰的看到上传的结果,我的网速太快了,我都没看清:

android retrofit2 设置 Params android retrofit上传文件_retrofit

不过确实是上传成功了,来看一下日志:

android retrofit2 设置 Params android retrofit上传文件_retrofit2实现表单文件上传_02

android retrofit2 设置 Params android retrofit上传文件_图片上传_03

android retrofit2 设置 Params android retrofit上传文件_图片上传_04

二、基本配置

既然是图文上传,这里重点讲的是上传,所以图片选择这一块我不打算多讲了,大家可以根据自己的实际业务去寻找对应的图片选择器(github上有很多),我这里使用的是MultiImageSelector,我把它的代码下载下来了,在本地项目中新建了一个Android Library,并给项目主Module添加上依赖,这样可以根据自己的需要进行定制,网络库这里肯定是选择Retrofit2.x版本了,直接添加对应的依赖到build.gradle文件中即可,结构如下图所示:

android retrofit2 设置 Params android retrofit上传文件_android retrofit_05

另外给大家推荐几款我觉得还不错的图片选择器的开源库,我这里因为没那么多需求,所以选了上面这个体积比较小的:

GalleryFinal:https://github.com/pengjianbo/GalleryFinal

PhotoPicker:https://github.com/donglua/PhotoPicker

Matisse(知乎开源的小清新款):https://github.com/zhihu/Matisse

三、代码实战

3.1、创建RetrofitManager和APIService

在最初的http协议中,没有定义上传文件的Method,为了实现这个功能,http协议组改造了post请求,添加了一种post规范,设定这种规范的Content-Type为multipart/form-data;boundary=bound,其中bound,其中{bound}是定义的分隔符,用于分割各项内容(文件,key-value对),不然服务器无法正确识别各项内容。post body里需要用到,尽量保证随机唯一。Retrofit是个网络代理框架,负责封装请求,然后把请求分发给http协议具体实现者-httpclient。retrofit默认的httpclient是okhttp。Retrofit会根据注解封装网络请求,待httpclient请求完成后,把原始response内容通过转化器(converter)转化成我们需要的对象(object)。那么Retrofit和okhttp怎么封装这些multipart/form-data上传数据呢?答案如下:

@retrofit2.http.Multipart: 标记一个请求是multipart/form-data类型,需要和@retrofit2.http.POST一同使用,并且方法参数必须是@retrofit2.http.Part注解。 

@retrofit2.http.Part: 代表Multipart里的一项数据,即用${bound}分隔的内容块。 
@retrofit2.http.PartMap:用于表单字段,默认接受的类型是Map<String,RequestBody>,可用于实现多文件上传。

了解了上面这些,就可以开始定义我们的ApiService了,注意这是个接口类,定义如下:

android retrofit2 设置 Params android retrofit上传文件_图片上传_06

返回值类型这里Retrofit返回的是Call类型,如果你是使用的RxJava,这里直接返回Observable就行了,泛型这里传入的是服务端返回内容的一个实体类,这样最后直接根据实体类中的字段去进行结果的处理就OK了!

定义完了接口,接下来我们再去定义一个Retrofit的处理类,当然你也可以不定义,直接根据Retrofit的官方api在用到的地方调用也行,但是实际项目中肯定是要封装一个类,不然要写多少遍啊,不符合我们CV工程师的特点了!关于Retrofit的用法不再细说了,你既然能看到这里了,肯定不是个小白了,直接上代码:

package com.nari.yihui.api;

import android.support.annotation.NonNull;
import android.util.Log;

import com.nari.yihui.BaseApplication;
import com.nari.yihui.commonlib.ui.smartshow.SmartToast;
import com.nari.yihui.constants.ApiConstant;
import com.nari.yihui.helper.UserManager;
import com.nari.yihui.utils.CommonUtil;

import java.io.IOException;
import java.util.concurrent.TimeUnit;

import okhttp3.Interceptor;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.logging.HttpLoggingInterceptor;
import retrofit2.Retrofit;
import retrofit2.adapter.rxjava2.RxJava2CallAdapterFactory;
import retrofit2.converter.gson.GsonConverterFactory;

/**
 * Created by 纪安奇 on 2018/6/11.
 * Retrofit的管理类
 */

@SuppressWarnings("FieldCanBeLocal")
public class RetrofitManager {

    //读超时时长,单位:毫秒
    public static final int READ_TIME_OUT = 15 * 1000;
    //写超时时长,单位:毫秒
    public static final int WRITE_TIME_OUT = 15 * 1000;
    //连接超时时长,单位:毫秒
    public static final int CONNECT_TIME_OUT = 15 * 1000;

    private static Retrofit mRetrofit; //Retrofit对象
    private static ApiService mApiService; //接口类实例对象
    private static OkHttpClient mClient; //OKHttp对象

    //获取ApiService对象
    public static ApiService getInstance() {
        if (CommonUtil.isNetworkAvailable(BaseApplication.getAppContext())) {
            //配置OkHttpClient
            OkHttpClient.Builder builder = new OkHttpClient.Builder();
            builder.readTimeout(READ_TIME_OUT, TimeUnit.MILLISECONDS)
                    .writeTimeout(WRITE_TIME_OUT, TimeUnit.MILLISECONDS)
                    .connectTimeout(CONNECT_TIME_OUT, TimeUnit.MILLISECONDS)
                    .addInterceptor(getLoggerInterceptor());
            //添加请求头Header
            if (UserManager.getInstance().getUser() != null) {
                builder.addInterceptor(new Interceptor() {
                    @Override
                    public Response intercept(@NonNull Chain chain) throws IOException {
                        Request request = chain.request()
                                .newBuilder()
                                .addHeader("Cookie", "sid=" + UserManager.getInstance().getUser().getObj().getSid())
                                .addHeader("Content-Type", "text/html;charset:utf-8")
                                .build();
                        return chain.proceed(request);
                    }
                });
            }
            mClient = builder.build();
            //配置Retrofit
            mRetrofit = new Retrofit.Builder()
                    .baseUrl(ApiConstant.baseUrl)
                    .addConverterFactory(GsonConverterFactory.create())
                    .addCallAdapterFactory(RxJava2CallAdapterFactory.create())
                    .client(mClient)
                    .build();
            //创建ApiService对象
            mApiService = mRetrofit.create(ApiService.class);
        } else {
            SmartToast.showInCenter("当前网络连接失败");
        }
        return mApiService;
    }

    //设置日志拦截器,打印返回的数据
    private static HttpLoggingInterceptor getLoggerInterceptor() {
        //日志显示级别
        HttpLoggingInterceptor.Level level = HttpLoggingInterceptor.Level.BODY;
        //新建log拦截器
        HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(new HttpLoggingInterceptor.Logger() {
            @Override
            public void log(@NonNull String message) {
                Log.e("ApiUrl------>", message);
            }
        });
        loggingInterceptor.setLevel(level);
        return loggingInterceptor;
    }

}

3.2、准备好选择的图片

既然是图片文字上传,首先我们肯定得准备好图片和文字,上面已经大致说了怎么去实现图片的选择,这里不会详细的说,因为篇幅有限,所以只讲如何上传。首先来看一下服务端需要我们如何拼接参数去提交的呢?

android retrofit2 设置 Params android retrofit上传文件_android retrofit_07

可以看到输入的意见文字字段为content,它需要我们和其它两部分内容一起拼接成一个json串然后提交,图片部分需要我们按照key-value的形式进行提交,key就是file,value就是我们对应的图片,有几张图片就有几组file=?,然后这两大部分一起构造成最后需要提交的参数。在我的项目中我把最后需要上传的图片文件都存放到了一个File[]类型的数组中,那么根据服务端的要求也就会写成(file,file[0])这种成对的形式。

3.3、开始构造参数

有了上面的思路之后,就开始干吧。我在本地创建了一个UploadHelper类,初步的构想是将这个类采用单例模式的形式,然后对于参数这部分采用建造者模式,这样方便参数的添加,后面我会给出这个类的代码,这里就先截图了:

整个类采用单例模式:

android retrofit2 设置 Params android retrofit上传文件_图片上传_08

对于参数部分,采用建造者模式进行构建:

android retrofit2 设置 Params android retrofit上传文件_retrofit2实现表单文件上传_09

我这里对于Map的value类型做了一个判断,文本和图片会根据对应的类型添加到Map中,是不是很方便,现在可以去真正的构造参数了,请看代码:

首先构造json串:

String jsonString = "{\"name\":\"" + name + "\",\"content\":\"" + content +
                            "\",\"province\":\"" +
                            BaseApplication.getSpUtil().getString(SPConstant.SP_PROVINCE) + "\"}";

接着构造图片,并且合并上面的json串:

helper = UploadHelper.getInstance();
        if (bmp.size() == 1) {
            helper.addParameter("file", files[0]);
        } else if (bmp.size() == 2) {
            helper.addParameter("file", files[0]);
            helper.addParameter("file", files[1]);
        } else if (bmp.size() == 3) {
            helper.addParameter("file", files[0]);
            helper.addParameter("file", files[1]);
            helper.addParameter("file", files[2]);
        }
        Map<String, RequestBody> params = helper.addParameter("param", json).builder();

3.4、实现上传

最后是发送我们的Http请求,实现上传的功能,这里根据处理结果做一些UI上的更新:

LoadingDialog.show(this, "正在上传,请稍候...", false);
        ApiConstant.baseUrl = ApiConstant.getBaseUrl(BaseUrlCode.FEEDBACK_UPLOAD);
        RetrofitManager.getInstance().uploadFeedback(params).enqueue(new Callback<CommonJsonModel>() {
            @Override
            public void onResponse(@NonNull Call<CommonJsonModel> call, @NonNull Response<CommonJsonModel> response) {
                if (response.isSuccessful()) {
                    LoadingDialog.canceled();
                    if (response.body() != null) {
                        if (response.body().getCode() == 1000) {
                            SmartToast.showInCenter("上传成功");
                            finish();
                            overridePendingTransition(R.anim.in_from_right, R.anim.finish_to_right);
                        } else if (response.body().getCode() == 2004 || response.body().getCode() == 2006) {
                            SmartToast.showInCenter("登录失效,请重新登录");
                            CommonUtil.loginOut(FeedBackActivity.this);
                        } else {
                            SmartToast.showInCenter(response.body().getMsg());
                        }
                    }
                }
            }

            @Override
            public void onFailure(@NonNull Call<CommonJsonModel> call, @NonNull Throwable t) {
                LoadingDialog.canceled();
                LogUtil.e("意见反馈上传图文异常------" + t.getMessage());
            }
        });

以上就是整个图文上传的实现过程,如果有不对的地方,还请告知,谢谢大家!

附:UploadHelper.java源码

package com.nari.yihui.helper;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Intent;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Environment;
import android.provider.MediaStore;

import com.nari.yihui.commonlib.imageselector.MultiImageSelectorActivity;
import com.nari.yihui.constants.Constant;
import com.nari.yihui.utils.FileUtils;
import com.nari.yihui.utils.LogUtil;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import okhttp3.MediaType;
import okhttp3.RequestBody;

/**
 * 包名:com.nari.yihui.helper
 * 文件名:   UploadHelper
 * 创建时间:   2018/7/19 18:50
 * 作者:     纪安奇
 * 作用:图文上传工具类
 */
public class UploadHelper {
    private volatile static UploadHelper mInstance;
    public static Map<String, RequestBody> params;
    private Uri photoUri;

    private UploadHelper() {}

    //单例模式
    public static UploadHelper getInstance() {
        if (mInstance == null) {
            synchronized (UploadHelper.class) {
                if (mInstance == null)
                    mInstance = new UploadHelper();
                params = new HashMap<>();
            }
        }
        return mInstance;
    }

    //根据传进来的Object对象来判断是String还是File类型的参数
    public UploadHelper addParameter(String key, Object o) {
        RequestBody body = null;
        if (o instanceof String) {
            body = RequestBody.create(MediaType.parse("text/plain;charset=UTF-8"), (String) o);
        } else if (o instanceof File) {
            body = RequestBody.create(MediaType.parse("multipart/form-data;charset=UTF-8"), (File) o);
        }
        params.put(key, body);
        return this;
    }

    //建造者模式
    public Map<String, RequestBody> builder() {
        return params;
    }

    //清除参数
    public void clear(){
        params.clear();
    }

    //最终上传的Bitmap保存为File对象
    public void saveBitmapFile(List<Bitmap> mList,File[] files) {
        String root = Environment.getExternalStorageDirectory().toString();
        File myDir = new File(root + "/suggestionUpload");
        if (myDir.exists()) {
            myDir.delete();
        }
        myDir.mkdirs();
        for (int i = 0; i < mList.size(); i++) {
            files[i] = new File(myDir, "ims" + i + ".JPEG");
            try {
                if (files[i].exists()) {
                    files[i].delete();
                }
                files[i].createNewFile();
                FileOutputStream out = new FileOutputStream(files[i]);
                mList.get(i).compress(Bitmap.CompressFormat.JPEG, 100, out);
                out.flush();
                out.close();
                LogUtil.e("最终上传图片的路径------>" + files[i].getAbsolutePath());
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    //启用裁剪
    public void startPhotoZoom(Activity mContext, List<String> drr) {
        try {
            // 获取系统时间 然后将裁剪后的图片保存至指定的文件夹
            @SuppressLint("SimpleDateFormat")
            SimpleDateFormat sDateFormat = new SimpleDateFormat("yyyyMMddhhmmss");
            String address = sDateFormat.format(new Date());
            if (!FileUtils.isFileExist("")) {
                FileUtils.createSDDir("");
            }
            drr.add(FileUtils.SDPATH + address + ".JPEG");
            @SuppressLint("SdCardPath")
            Uri imageUri = Uri.parse("file:///sdcard/formats/" + address + ".JPEG");
            final Intent intent = new Intent("com.android.camera.action.CROP");
            // 照片URL地址
            intent.setDataAndType(photoUri, "image/*");
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", 480);
            intent.putExtra("outputY", 480);
            // 输出路径
            intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
            // 输出格式
            intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
            // 不启用人脸识别
            intent.putExtra("noFaceDetection", false);
            intent.putExtra("return-data", false);
            mContext.startActivityForResult(intent, Constant.REQUEST_CUTTING_CODE);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    //拍照
    public void takePhoto(Activity mContext) {
        try {
            Intent openCameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            String sdcardState = Environment.getExternalStorageState();
            String sdcardPathDir = Environment.getExternalStorageDirectory().getPath() + "/myImage/";
            File file = null;
            if (Environment.MEDIA_MOUNTED.equals(sdcardState)) {
                // 有sd卡,是否有myImage文件夹
                File fileDir = new File(sdcardPathDir);
                if (!fileDir.exists()) {
                    fileDir.mkdirs();
                }
                // 是否有headImg文件
                file = new File(sdcardPathDir + System.currentTimeMillis() + ".JPEG");
            }
            if (file != null) {
                String path = file.getPath();
                photoUri = Uri.fromFile(file);
                openCameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
                mContext.startActivityForResult(openCameraIntent, Constant.REQUEST_CAMERA_CODE);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    //从相册选择
    public void albmSelect(Activity mContext,int maxCanSelectNum,List<String> drr) {
        int selectedMode = MultiImageSelectorActivity.MODE_MULTI;
        int maxNum = maxCanSelectNum - drr.size();
        Intent intent = new Intent(mContext, MultiImageSelectorActivity.class);
        // 是否显示拍摄图片
        intent.putExtra(MultiImageSelectorActivity.EXTRA_SHOW_CAMERA, false);
        // 最大可选择图片数量
        intent.putExtra(MultiImageSelectorActivity.EXTRA_SELECT_COUNT, maxNum);
        // 选择模式
        intent.putExtra(MultiImageSelectorActivity.EXTRA_SELECT_MODE, selectedMode);
        mContext.startActivityForResult(intent, Constant.REQUEST_ALBUM_CODE);
    }

}