网络一直是我个人的盲点,前一阵子抽出时间学习了一下Volley网络工具的使用方法,也透过源码进行了进一步的学习,有一些心得想分享出来。在Android开发中,成熟的网络工具不少,Android自带了HttpClient,还有okhttp,还有koush大神创建的ion开源项目,然后就是google后来加入到Android项目源码中的Volley。为什么使用Volley,是因为Volley使用简单,逻辑清晰,即使在调试过程中出现了问题,也可以快速的通过源码进行定位。
因为已经习惯了使用Gradle构架应用,所以我在第一次想要使用Volley的时候尝试寻找是否可以通过gradle的配置文件进行库依赖。可惜的是,并没有。但即使这样Volley的库也很容易做出来加入到我们的工程中。
首先需要ant编译工具,然后如果有Android系统源码的话,Volley在frameworks/volley目录下。如果没有Android源码,也很好办,可以单独从Android的仓储中克隆出Volley源码:
git clone https://android.googlesource.com/platform/frameworks/volley
不幸的是,volley库的源码Android并没有托管在其在Github的帐号上,所以只能在googlesource上进行克隆,当然在国内也就需要先FQ才可以了。
下图为Volley源码结构:
克隆成功后,可以方便的使用ant进行编译,当然,如果是在完整的Android源码下,也可以直接通过make进行编译,但是时间必然会长很多。这里使用ant编译为例,执行:
ant jar
结果如图所示:
这样jar包就生成了,很方便吧,接下来将其添加到工程中就可以使用了。
Volley的网络请求父类为Request
为了更加贴近实际使用,下边将使用Volley与Cloudant进行通讯做示例。Cloudant是一家提供云服务业务的公司,其向开发者提供免费的云存储、云数据库服务。关于其注册等流程本文不做叙述,很简单的。直接从登录开始:
Volley的一个很大的特色,就是所有的网络请求无需开发者自己执行,而是在请求构造完成后扔到Volley的请求队列中,队列依次进行请求,这样就省去了很多麻烦,开发者也不用担心网络请求是否会冲突,是否会在主线程,这些烦心事Volley的网络队列都帮我们解决了。
一般来说,一个应用程序如果网络请求没有特别频繁则完全可以只有一个请求队列(对应Application),如果非常多或其他情况,则可以是一个Activity对应一个网络请求队列,具体情况具体分析。下边的代码展示了如何申请一个Volley网络请求队列:
RequestQueue mQueue;
mQueue = Volley.newRequestQueue(getApplicationContext());
这样就成功申请了一个网络请求队列,如果只有一个,则可以在Application中进行申请。
假设已经成功注册,登录名foo,密码bar。
通过查阅Cloudant的登录认证文档:https://docs.cloudant.com/api/authn.html。可以发现Cloudant登录认证相关接口有三个:
这里我们使用POST方法进行cookie登录认证。结合上边假设的用户名和密码可知:
要访问的url为 foo.cloudant.com/_session
头信息为 Content-Type: application/x-www-form-urlencoded
参数为 name = foo, password = bar
若访问成功,我们就可以在网络回应中获取cookie,以备之后其他操作使用。显然,这个请求跟json毫无关系,应该使用StringRequest,StringRequest有两种构造方法:
public StringRequest(int method, String url, Listener listener, ErrorListener errorListener)
public StringRequest(String url, Listener listener, ErrorListener errorListener)
第二个方法只有GET请求才可以使用,第一个方法的method参数可以用来自定义请求类型,这里我们需要的是POST,所以应该使用第一个构造方法:
StringRequest request = new StringRequest(
Request.Method.POST,
"http://foo.cloudant.com/_session",
new Response.Listener() {
@Override
public void onResponse(String s) { //收到成功应答后会触发这里
}
},
new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) { //出现连接错误会触发这里
}
}
);
上边的代码中,我们成功构造了一个StringRequest,其中已经包含了我们需要的POST和正确的URL,同时还添加了网络回应监听器。但是,还缺少文档要求我们的头信息和参数。StringRequest在构造中并不提供这些信息的定义,这也是与其他常用网络工具不同的地方,刚接触的同学可能会很不适用,通过复写StringRequest的两个方法就可以将这些信息放进去了。下边来完善这个请求:
StringRequest request = new StringRequest(
Request.Method.POST,
"http://foo.cloudant.com/_session",
new Response.Listener() {
@Override
public void onResponse(String s) {
}
},
new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
}
}
) {
@Override
public Map getHeaders() throws AuthFailureError { //设置头信息
Map map = new HashMap();
map.put("Content-Type", "application/x-www-form-urldecoded");
return map;
}
@Override
protected Map getParams() throws AuthFailureError { //设置参数
Map map = new HashMap();
map.put("name", "foo");
map.put("password", "bar");
return map;
}
};
相比第一次我们的构造过程,这一次多了两个复写的方法来设置头信息和参数,很容易吧。这个时候请求基本完成了,但是却缺少另一个很重要的东西,我们的登录认证为的是拿回属于自己的cookie,如果不能获取cookie的话,多么正确的请求格式都是白费力气啊,想要拿到cookie一样也是通过复写另一个方法进行获取:
@Override
protected Response parseNetworkResponse(NetworkResponse response) {
for (String s : response.headers.keySet()) {
if (s.contains("Set-Cookie")) {
mCookie = response.headers.get(s);
break;
}
}
return super.parseNetworkResponse(response);
}
在网络请求成功后,服务端返回应答信息,而我们所需的Cookie信息就在这些应答信息中,通过对应答信息的遍历查找,很方便就可以找到我们所需的信息了。到这里,我们的登录认证请求就构造完成了,最后需要做的就是将这个StringRequest扔到我们的请求队列中去:
mQueue.add(request);
网络通畅的情况下,很快就能够获取Cookie信息了。
在注册Cloudant成功后,Cloudant会在我们的帐号中创建一个默认数据库——crud,其中保存着一行测试数据welcome。
让我们用Volley来访问这条数据。查阅Cloudant API文档Documents相关可以发现:
通过简单的GET请求搭配正确的URL即可得到文件(数据)内容,当然,这一切的前提是我们已经掌握了正确的Cookie数据。那么,我们需要:
1. 请求头数据中包含正确的Cookie信息
2. 访问正确的URL
3. 请求类型:GET
假设通过上一步登陆认证后我们将Cookie信息保存在了mCookie字符串变量中。而我们需要访问的URL通过查阅文档也可以得出路径为 数据库名 + 文档名,即foo.cloudant.com/crud/welcome。万事俱备,使用StringRequest:
StringRequest request = new StringRequest(
"http://foo.cloudant.com/crud/welcome",
new Response.Listener() {
@Override
public void onResponse(String s) {
}
},
new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
}
}
) {
@Override
public Map getHeaders() throws AuthFailureError {
Map map = new HashMap();
map.put("Cookie", mCookie);
return map;
}
};
mQueue.add(request);
简单的网络请求StringRequest完全处理得来,使用也比较简单,就介绍到这里。下边介绍JsonObjectRequest应用方法。
public JsonObjectRequest(int method, String url, JSONObject jsonRequest, Listener listener, ErrorListener errorListener)
public JsonObjectRequest(String url, JSONObject jsonRequest, Listener listener, ErrorListener errorListener)
第一种方法参数以此为:请求方法,访问的URL,Json数据对象,请求成功监听器,请求失败监听器。
1. 访问的URL path为数据库目录
2. Content-Type被要求为application/json
3. 携带的数据要求为json数据
既然方法要求为POST,我们又是创建数据,肯定数据内容不会为空,所以我们选择第二种构造方法。首先,创建一个Json对象:
JSONObject jsonObject = new JSONObject();
jsonObject.put("_id", "testinfo");
jsonObject.put("person", "foo");
jsonObject.put("phone", "bar");
JsonObjectRequest request = new JsonObjectRequest(
"http://foo.cloudant.com/crud",
jsonObject,
new Response.Listener() {
@Override
public void onResponse(JSONObject jsonObject) {
}
},
new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
}
}
) {
@Override
public Map getHeaders() throws AuthFailureError {
Map map = new HashMap();
map.put("Cookie", mCookie);
return map;
}
};
mQueue.add(request);
@Override
public String getBodyContentType() {
return PROTOCOL_CONTENT_TYPE;
}
/** Charset for request. */
private static final String PROTOCOL_CHARSET = "utf-8";
/** Content type for request. */
private static final String PROTOCOL_CONTENT_TYPE =
String.format("application/json; charset=%s", PROTOCOL_CHARSET);
关于Volley和Cloudant更多的通信细节,见CloudantVolley项目:https://github.com/airk000/CloudantVolley