[Android] 解决Volley中JsonObjectRequest的Post请求添加参数的问题

由于一开始官方介绍 Volley 适合轻量高并发的网络请求场景, 并不推荐用于上传下载, 因此以前只是粗略了解下就浅尝辄止, 并没有在项目中正式使用. 直到最近用到Volley.  于是碰到了一个问题.


使用 JsonObjectRequest  发POST请求时, Volley官方说在getParams(xxx) 方法中传递POST参数是无效的, 需要在构造方法中通过 JsonObject 去传递参数, 类似这样:

Map params = new HashMap();
params.put("username","hello_world");
params.put("password", "123456");
params.put("sex","1");

JSONObject paramJsonObject = new JSONObject(params);

JsonObjectRequest mJsonObjectRequest = new JsonObjectRequest(Request.Method.POST,
                url,paramJsonObject,
                successListener, errorListener){
					//...
				
				};

但是后台并不会正常接收到, 除非后台是JsonObject格式去接收并处理客户端这边发出的请求参数. 因为传递到后台的参数是这样的:

{"username":"hello_world","password":"123456","sex":"1"}


我们直接看Volley中 JsonObjectRequest  构造方法的相关源码:

public JsonObjectRequest(int method, String url, JSONObject jsonRequest,
            Listener listener, ErrorListener errorListener) {
			
        super(method, url, (jsonRequest == null) ? null : jsonRequest.toString(), listener,
                errorListener);
				
}
上面代码可看出, 是直接将装载参数的 JsonObject进行 toString() 处理, 出来的结果就是我们上面所看到的那样. 最后还是回调

public JsonRequest(int method, String url, String requestBody, Listener listener,
            ErrorListener errorListener) {
			
        super(method, url, errorListener);
        mListener = listener;
        mRequestBody = requestBody;
		
}
这个方法, 将参数传到String类型的 requestBody, 实现真正的参数传递.


所以结论是, 直接在 JsonObject 中放参数, 是无效的. 而网上到处传来传去说 JsonObjectRequest 请求中在 getParams(...) 中传参无效, getParams(...) 只针对StringRequest. 但也没有给出实际的方案, 还推荐用JsonObject传参...


其实追溯源码看看 JsonObjectRequest, StringRequest 和 Request 中几个类的关键方法, 我们有2种方法在 JsonObjectRequest 的 POST 请求中传参.

第一种: requestBody

从上面的构造方法看出JsonObject最后还是要toString() 并传给requestBody, 而且 JsonObjectRequest 本身也提供了一个包含 requestBody 的构造方法. 因此可以仿效GET请求的格式这样实现传参:

String requestBody = "username=hello_world&password=123456&sex=1";

JsonObjectRequest mJsonObjectRequest = new JsonObjectRequest	
(Request.Method.POST,
                url,requestBody,
                successListener, errorListener){
				
				// ...
				
};

第二种: getBody(...) 方法

这种方法跟 requestBody 是有关系的. 因为 requestBody 到最后是传值给 mRequestBody. 而 mRequestBody 则在 JsonRequest 中的 getBody(...) 方法被使用:

@Override
public byte[] getPostBody() {
    return getBody();
}

@Override
public byte[] getBody() {
    try {
        return mRequestBody == null ? null : mRequestBody.getBytes(PROTOCOL_CHARSET);
    } catch (UnsupportedEncodingException uee) {
        VolleyLog.wtf("Unsupported Encoding while trying to get the bytes of %s using %s",
                mRequestBody, PROTOCOL_CHARSET);
        return null;
    }
}
一切参数在 getPostBody() 中被返回, 很明显, 这个方法才是最终返回POST参数给请求的.


如果到了这里还不明白, 整个传参的过程, 请直接看 Request 类的这几个方法:

/**
 * Returns the raw POST body to be sent.
 *
 * @throws AuthFailureError In the event of auth failure
 *
 * @deprecated Use {@link #getBody()} instead.
 */
@Deprecated
public byte[] getPostBody() throws AuthFailureError {
	// Note: For compatibility with legacy clients of volley, this implementation must remain
	// here instead of simply calling the getBody() function because this function must
	// call getPostParams() and getPostParamsEncoding() since legacy clients would have
	// overridden these two member functions for POST requests.
	Map postParams = getPostParams();
	if (postParams != null && postParams.size() > 0) {
		return encodeParameters(postParams, getPostParamsEncoding());
	}
	return null;
}

/**
 * Returns the raw POST or PUT body to be sent.
 *
 * 

By default, the body consists of the request parameters in * application/x-www-form-urlencoded format. When overriding this method, consider overriding * {@link #getBodyContentType()} as well to match the new body format. * * @throws AuthFailureError in the event of auth failure */ public byte[] getBody() throws AuthFailureError { Map params = getParams(); if (params != null && params.size() > 0) { return encodeParameters(params, getParamsEncoding()); } return null; } /** * Returns a Map of POST parameters to be used for this request, or null if * a simple GET should be used. Can throw {@link AuthFailureError} as * authentication may be required to provide these values. * *

Note that only one of getPostParams() and getPostBody() can return a non-null * value.

* @throws AuthFailureError In the event of auth failure * * @deprecated Use {@link #getParams()} instead. */ @Deprecated protected Map getPostParams() throws AuthFailureError { return getParams(); } /** * Returns a Map of parameters to be used for a POST or PUT request. Can throw * {@link AuthFailureError} as authentication may be required to provide these values. * *

Note that you can directly override {@link #getBody()} for custom data.

* * @throws AuthFailureError in the event of auth failure */ protected Map getParams() throws AuthFailureError { return null; } /** * Converts params into an application/x-www-form-urlencoded encoded string. */ private byte[] encodeParameters(Map params, String paramsEncoding) { StringBuilder encodedParams = new StringBuilder(); try { for (Map.Entry entry : params.entrySet()) { encodedParams.append(URLEncoder.encode(entry.getKey(), paramsEncoding)); encodedParams.append('='); encodedParams.append(URLEncoder.encode(entry.getValue(), paramsEncoding)); encodedParams.append('&'); } return encodedParams.toString().getBytes(paramsEncoding); } catch (UnsupportedEncodingException uee) { throw new RuntimeException("Encoding not supported: " + paramsEncoding, uee); } }
特别是 encodeParameters(...) 方法, 完全看到无论是JsonObject 还是 MAP 装着数据传进来, 最后都是加工成类似 GET 的 "key=value" 的格式. 同时, 我们也明白 StringRequest 如何通过 getParams() 传参的.


这样就好办了, 所以我们在 JsonObjectRequest 的构造方法里直接重写 getBody() 方法就行了:

final String requestBody = "username=hello_world&password=123456&sex=1";

JsonObjectRequest req = new JsonObjectRequest(Request.Method.POST,
                url,
                successListener, errorListener) {

	@Override
	public byte[] getBody() {

		try {
			return requestBody.toString().getBytes("UTF-8");
			
		} catch (Exception e) {
		}
		return null;
	}
};

















你可能感兴趣的:(安卓开发)