volleyを使って画像をアップロードして、1つのパラメータの複数の図、複数の画像の複数の図、親測は有効です

5419 ワード

一、前言
Googleは2013のIO大会でvolleyフレームワークを発表してから広く応用されていますが、確かに、いくつかのネットリクエストライブラリを使ったことがありますが、volleyは使いやすく、使いやすいと思います.しかし、ファイルをアップロードするのは面倒です.特に、パラメータ名が複数のファイルに対応していると思います.私の父のバックグラウンドがくれたインタフェースのように、パラメータのkeyをimagesと呼び、valueはマルチマップです.多図...図....
多くのネットワークリクエストライブラリのpostリクエストがある場合、パラメータ付きでhashmapを使用してカプセル化されますが、hashmapはキー値対の形で値を格納します.ここで1つの問題は、keyが1つのvalueに対応し、2つのkeyが連続してputに入ると、後者が前者に取って代わることです.最後にパラメータが1つしか残っていません.これでは需要に達しない.1つの解決策は、複数のサブスレッドを開いて、毎回1枚の画像を転送することです.実は、これでもいいです.具体的には、需要を見て、私がここで議論するのは、一度に複数の図を転送することです.
二、解決
volleyを使うと、リクエストデータはどれも簡単ですが、ファイルのアップロードが面倒なのはよくありません.同時に複数のネットワークリクエストクラスライブラリを使うのもあまりお勧めできません.そこでここでは、httpmime(ダウンロードをクリック)を使用し、主にMultipartEntityクラスを使用しており、リクエストパラメータをカプセル化することができます.
主にvolleyのRequestクラスを継承し、httpmimのMultipartEntityクラスを使用してファイルパラメータをカプセル化します.ここでは、パラメータ名が1つのファイルに対応することを実現します.
1つのパラメータ名が複数のファイルに対応しています.複数のパラメータ名が複数のファイルに対応している場合は、自分で実装してみてください.
package com.android.volley.toolbox;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.http.entity.mime.MultipartEntity;
import org.apache.http.entity.mime.content.FileBody;
import org.apache.http.entity.mime.content.StringBody;

import com.android.volley.AuthFailureError;
import com.android.volley.NetworkResponse;
import com.android.volley.Request;
import com.android.volley.Response;
import com.android.volley.VolleyLog;
import com.common.utils.CLog;
import com.common.utils.FileUtil;

public class MultipartRequest extends Request<String> {

	private MultipartEntity entity = new MultipartEntity();

	private final Response.Listener<String> mListener;

	private List<File> mFileParts;
	private String mFilePartName;
	private Map<String, String> mParams;
	/**
	 *     
	 * @param url
	 * @param errorListener
	 * @param listener
	 * @param filePartName
	 * @param file
	 * @param params
	 */
	public MultipartRequest(String url, Response.ErrorListener errorListener,
			Response.Listener<String> listener, String filePartName, File file,
			Map<String, String> params) {
		super(Method.POST, url, errorListener);

		mFileParts = new ArrayList<File>();
		if (file != null) {
			mFileParts.add(file);
		}
		mFilePartName = filePartName;
		mListener = listener;
		mParams = params;
		buildMultipartEntity();
	}
	/**
	 *     ,    key
	 * @param url
	 * @param errorListener
	 * @param listener
	 * @param filePartName
	 * @param files
	 * @param params
	 */
	public MultipartRequest(String url, Response.ErrorListener errorListener,
			Response.Listener<String> listener, String filePartName,
			List<File> files, Map<String, String> params) {
		super(Method.POST, url, errorListener);
		mFilePartName = filePartName;
		mListener = listener;
		mFileParts = files;
		mParams = params;
		buildMultipartEntity();
	}

	private void buildMultipartEntity() {
		if (mFileParts != null && mFileParts.size() > 0) {
			for (File file : mFileParts) {
				entity.addPart(mFilePartName, new FileBody(file));
			}
			long l = entity.getContentLength();
			CLog.log(mFileParts.size()+" ,  :"+l);
		}

		try {
			if (mParams != null && mParams.size() > 0) {
				for (Map.Entry<String, String> entry : mParams.entrySet()) {
					entity.addPart(
							entry.getKey(),
							new StringBody(entry.getValue(), Charset
									.forName("UTF-8")));
				}
			}
		} catch (UnsupportedEncodingException e) {
			VolleyLog.e("UnsupportedEncodingException");
		}
	}

	@Override
	public String getBodyContentType() {
		return entity.getContentType().getValue();
	}

	@Override
	public byte[] getBody() throws AuthFailureError {
		ByteArrayOutputStream bos = new ByteArrayOutputStream();
		try {
			entity.writeTo(bos);
		} catch (IOException e) {
			VolleyLog.e("IOException writing to ByteArrayOutputStream");
		}
		return bos.toByteArray();
	}

	@Override
	protected Response<String> parseNetworkResponse(NetworkResponse response) {
		CLog.log("parseNetworkResponse");
		if (VolleyLog.DEBUG) {
			if (response.headers != null) {
				for (Map.Entry<String, String> entry : response.headers
						.entrySet()) {
					VolleyLog.d(entry.getKey() + "=" + entry.getValue());
				}
			}
		}

		String parsed;
		try {
			parsed = new String(response.data,
					HttpHeaderParser.parseCharset(response.headers));
		} catch (UnsupportedEncodingException e) {
			parsed = new String(response.data);
		}
		return Response.success(parsed,
				HttpHeaderParser.parseCacheHeaders(response));
	}


	/*
	 * (non-Javadoc)
	 * 
	 * @see com.android.volley.Request#getHeaders()
	 */
	@Override
	public Map<String, String> getHeaders() throws AuthFailureError {
		VolleyLog.d("getHeaders");
		Map<String, String> headers = super.getHeaders();

		if (headers == null || headers.equals(Collections.emptyMap())) {
			headers = new HashMap<String, String>();
		}


		return headers;
	}

	@Override
	protected void deliverResponse(String response) {
		mListener.onResponse(response);
	}
}

このリクエストで複数のファイルをアップロードでき、多くの解釈もせず、コードも簡単です.私もこの方法をテストしましたが、サーバーと付き合っているので、ここには確かに何も展示できませんが、説得力のある説明が好きです.