Android:Volley的使用及其工具类的封装
Android:Volley的使用及其工具类的封装
Android 2015-12-27
一. Volley简介
Volley的中文翻译为“齐射、并发”,是在2013年的Google大会上发布的一款Android平台网络通信库,具有网络请求的处理、小图片的异步加载和缓存等功能,能够帮助 Android APP 更方便地执行网络操作,而且更快速高效。
在Google IO的演讲上,其配图是一幅发射火弓箭的图,有点类似流星。这表示,Volley特别适合数据量不大但是通信频繁的场景。见下图:
Volley 有如下的优点:
- 自动调度网络请求;
- 高并发网络连接;
- 通过标准的 HTTP cache coherence(高速缓存一致性)缓存磁盘和内存透明的响应;
- 支持指定请求的优先级;
- 网络请求cancel机制。我们可以取消单个请求,或者指定取消请求队列中的一个区域;
- 框架容易被定制,例如,定制重试或者回退功能;
- 包含了调试与追踪工具;
Volley 不适合用来下载大的数据文件。因为 Volley 会保持在解析的过程中所有的响应。对于下载大量的数据操作,请考虑使用 DownloadManager。
在volley推出之前我们一般会选择比较成熟的第三方网络通信库,如:android-async-http、retrofit、okhttp等。他们各有优劣,可有所斟酌地选择选择更适合项目的类库。
附录:
Volley的github地址:https://github.com/mcxiaoke/android-volley;
Google I/O 2013 – Volley: Easy, Fast Networking for Android:https://www.youtube.com/watch?v=yhv8l9F44qo&feature=player_embedded
二. Volley jar包的导入
Volley 框架的核心代码是托管在 AOSP 仓库的 frameworks/volley 中,相关的工具放在 toolbox 下。
把 Volley 添加到项目中最简便的方法是 Clone 仓库,然后把它设置为一个 library project。
1) clone代码:
git clone https://android.googlesource.com/platform/frameworks/volley
2)将代码编译成jar包:
android update project -p . ant jar
如无意外,将获得volley.jar包。
3)添加volley.jar到你的项目中:
可参考:http://jingyan.baidu.com/article/e6c8503c7190b7e54f1a1893.html
备注:
附上我的volley.jar包的地址:http://pan.baidu.com/s/1sjSwCrV,方便大家直接下载使用。当然,Volley更新较快,还是希望大家能直接通过clone代码后进行编译。
三. Volley框架的详细使用:
Volley工作原理图如下:
使用Volley框架实现网络数据请求主要有以下三个步骤:
- 1.创建RequestQueue对象,定义网络请求队列;
- 2.创建XXXRequest对象(XXX代表String,JSON,Image等等),定义网络数据请求的详细过程;
- 3.把XXXRequest对象添加到RequestQueue中,开始执行网络请求。
3.1 创建RequestQueue对象
一般而言,网络请求队列都是整个APP内使用的全局性对象,因此最好写入Application类中:
public class MyApplication extends Application{
// 建立请求队列
public static RequestQueue queue;
@Override
public void onCreate() {
super.onCreate();
queue = Volley.newRequestQueue(getApplicationContext());
}
public static RequestQueue getHttpQueue() {
return queue;
}
}
这是,我们还需要修改AndroidManifest.xml文件,使APP的Application对象为我们刚定义的MyApplication,并添加INTERNET权限:
<uses-permission android:name="android.permission.INTERNET" />
<application
android:name=".MyApplication"
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme" >
</application>
3.2 创建XXXRequest对象并添加到请求队列中
Volley提供了JsonObjectRequest、JsonArrayRequest、StringRequest等Request形式:
- JsonObjectRequest:返回JSONObject对象;
- JsonArrayRequest:返回JsonArray对象;
- StringRequest:返回String。
另外可以继承Request自定义Request。
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// GET请求
VolleyGet();
// POST请求
VolleyPost();
}
// 定义POST请求的方法
private void VolleyPost() {
// 请求地址
String url = "http://ce.sysu.edu.cn/hope/";
// 创建StringRequest,定义字符串请求的请求方式为POST,
StringRequest request = new StringRequest(Request.Method.POST, url, new Response.Listener<String>() {
// 请求成功后执行的函数
@Override
public void onResponse(String s) {
// 打印出POST请求返回的字符串
Toast.makeText(MainActivity.this, "POST: " + s, Toast.LENGTH_LONG).show();
}
}, new Response.ErrorListener() {
// 请求失败时执行的函数
@Override
public void onErrorResponse(VolleyError volleyError) {
}
}){
// 定义请求数据
@Override
protected Map<String, String> getParams() throws AuthFailureError {
Map<String, String> hashMap = new HashMap<String, String>();
hashMap.put("phone", "11111");
return hashMap;
}
};
// 设置该请求的标签
request.setTag("abcPost");
// 将请求添加到队列中
MyApplication.getHttpQueue().add(request);
}
// 定义GET请求的方法
private void VolleyGet() {
// 定义请求地址
String url = "http://ce.sysu.edu.cn/hope/";
StringRequest request = new StringRequest(Request.Method.GET, url, new Response.Listener<String>() {
@Override
public void onResponse(String s) {
// 打印出GET请求返回的字符串
Toast.makeText(MainActivity.this, s, Toast.LENGTH_LONG).show();
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
}
});
// 设置该请求的标签
request.setTag("abcGet");
// 将请求添加到队列中
MyApplication.getHttpQueue().add(request);
}
}
输出:
// POST请求
POST: <厚朴网站首页源代码。。。>
// GET请求
GET: <厚朴网站首页源代码。。。>
3.3 关闭请求
3.3.1 关闭特定标签的网络请求:
// 网络请求标签为"abcGet"
public void onStop() {
super.onStop();
MyApplication.getHttpQueues.cancelAll("abcGet");
}
3.3.2 取消这个队列里的所有请求:
在activity的onStop()方法里面,取消所有的包含这个tag的请求任务。
@Override
protected void onStop() {
super.onStop();
mRequestQueue.cancelAll(this);
}
四. GET和POST请求工具库的封装
4.1 重写Application
因为网络请求队列相对于APP应用老说是全局对象,因此可以定义在全局中。为此,我们新建一个LIMSApplication,并让其继承自Application。
LIMSApplication.java文件:
public class LIMSApplication extends Application {
public static RequestQueue volleyQueue;
@Override
public void onCreate() {
super.onCreate();
/* Volley配置 */
// 建立Volley的Http请求队列
volleyQueue = Volley.newRequestQueue(getApplicationContext());
}
// 开放Volley的HTTP请求队列接口
public static RequestQueue getRequestQueue() {
return volleyQueue;
}
}
不要忘记在AndroidManifest.xml文件中修改Application的name和相应的网络请求权限:
<uses-permission android:name="android.permission.INTERNET" />
<application
android:name=".LIMSApplication"
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme" >
</application>
4.2 GET和POST请求的封装:
目前,VolleyRequestUtil工具库只包含了两个函数,分别获取GET和POST请求。
VolleyRequestUtil.java:
public class VolleyRequestUtil {
public static StringRequest stringRequest;
public static Context context;
/*
* 获取GET请求内容
* 参数:
* context:当前上下文;
* url:请求的url地址;
* tag:当前请求的标签;
* volleyListenerInterface:VolleyListenerInterface接口;
* */
public static void RequestGet(Context context, String url, String tag, VolleyListenerInterface volleyListenerInterface) {
// 清除请求队列中的tag标记请求
LIMSApplication.getRequestQueue().cancelAll(tag);
// 创建当前的请求,获取字符串内容
stringRequest = new StringRequest(Request.Method.GET, url, volleyListenerInterface.responseListener(), volleyListenerInterface.errorListener());
// 为当前请求添加标记
stringRequest.setTag(tag);
// 将当前请求添加到请求队列中
LIMSApplication.getRequestQueue().add(stringRequest);
// 重启当前请求队列
LIMSApplication.getRequestQueue().start();
}
/*
* 获取POST请求内容(请求的代码为Map)
* 参数:
* context:当前上下文;
* url:请求的url地址;
* tag:当前请求的标签;
* params:POST请求内容;
* volleyListenerInterface:VolleyListenerInterface接口;
* */
public static void RequestPost(Context context, String url, String tag, final Map<String, String> params, VolleyListenerInterface volleyListenerInterface) {
// 清除请求队列中的tag标记请求
LIMSApplication.getRequestQueue().cancelAll(tag);
// 创建当前的POST请求,并将请求内容写入Map中
stringRequest = new StringRequest(Request.Method.POST, url, volleyListenerInterface.responseListener(), volleyListenerInterface.errorListener()){
@Override
protected Map<String, String> getParams() throws AuthFailureError {
return params;
}
};
// 为当前请求添加标记
stringRequest.setTag(tag);
// 将当前请求添加到请求队列中
LIMSApplication.getRequestQueue().add(stringRequest);
// 重启当前请求队列
LIMSApplication.getRequestQueue().start();
}
}
4.3 Volley请求(成功或失败)的监听事件封装:
封装Volley请求(成功或失败)的监听事件,见VolleyListenerInterface.java:
public abstract class VolleyListenerInterface {
public Context mContext;
public static Response.Listener<String> mListener;
public static Response.ErrorListener mErrorListener;
public VolleyListenerInterface(Context context, Response.Listener<String> listener, Response.ErrorListener errorListener) {
this.mContext = context;
this.mErrorListener = errorListener;
this.mListener = listener;
}
// 请求成功时的回调函数
public abstract void onMySuccess(String result);
// 请求失败时的回调函数
public abstract void onMyError(VolleyError error);
// 创建请求的事件监听
public Response.Listener<String> responseListener() {
mListener = new Response.Listener<String>() {
@Override
public void onResponse(String s) {
onMySuccess(s);
}
};
return mListener;
}
// 创建请求失败的事件监听
public Response.ErrorListener errorListener() {
mErrorListener = new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
onMyError(volleyError);
}
};
return mErrorListener;
}
}
4.3 Volley图片加载库的封装:
Volley库还具有图片加载的功能。但适合小图片的异步加载,不适合于比较大的图片资源的请求。
Volley提供了多种Request方法,譬如ImageRequest、ImageLoader、NetWorkImageView。
网络图片资源的请求封装如下:
ImageLoaderUtil.java:
public class ImageLoaderUtil {
/*
* 通过ImageRequest来显示网络图片
* */
public static void setImageRequest(String url, final ImageView imageView) {
ImageRequest imageRequest = new ImageRequest(url, new Response.Listener<Bitmap>() {
@Override
public void onResponse(Bitmap bitmap) {
imageView.setImageBitmap(bitmap);
}
}, 0, 0, Bitmap.Config.RGB_565, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) {
imageView.setBackgroundResource(R.mipmap.ic_launcher);
}
});
LIMSApplication.getRequestQueue().add(imageRequest);
}
/*
* 通过ImageLoader来显示网络图片
* */
public static void setImageLoader(String url, ImageView imageView, int defaultImageResId, int errorImageResId) {
ImageLoader loader = new ImageLoader(LIMSApplication.getRequestQueue(), new BitmapCache());
ImageLoader.ImageListener imageListener = ImageLoader.getImageListener(imageView, defaultImageResId, errorImageResId);
loader.get(url, imageListener);
}
/*
* 通过Volley的NetWorkImageView来显示网络图片
* */
public static void setNetWorkImageView(String url, NetworkImageView netWorkImageView, int defaultImageResId, int errorImageResId) {
ImageLoader loader = new ImageLoader(LIMSApplication.getRequestQueue(), new BitmapCache());
netWorkImageView.setDefaultImageResId(defaultImageResId);
netWorkImageView.setErrorImageResId(errorImageResId);
netWorkImageView.setImageUrl(url, loader);
}
}
五. VolleyRequestUtil与ImageLoaderUtil的使用
5.1 用GET方式请求网络资源:
new VolleyRequestUtil().RequestGet(this, "http://ce.sysu.edu.cn/hope/", "hopePage",
new VolleyListenerInterface(this, VolleyListenerInterface.mListener, VolleyListenerInterface.mErrorListener) {
// Volley请求成功时调用的函数
@Override
public void onMySuccess(String result) {
Toast.makeText(this, s, Toast.LENGTH_LONG).show();
}
// Volley请求失败时调用的函数
@Override
public void onMyError(VolleyError error) {
// ...
}
});
输出:厚朴网站首页的源代码。
5.2 用POST方式请求网络资源:
new VolleyRequestUtil().RequestPOST(this, "http://ce.sysu.edu.cn/hope/", "hopePage",
new VolleyListenerInterface(this, VolleyListenerInterface.mListener, VolleyListenerInterface.mErrorListener) {
// Volley请求成功时调用的函数
@Override
public void onMySuccess(String result) {
Toast.makeText(MainActivity.this, result, Toast.LENGTH_LONG).show();
}
// Volley请求失败时调用的函数
@Override
public void onMyError(VolleyError error) {
// ...
}
});
输出:厚朴网站首页的源代码。
5.3 通过ImageRequest来显示网络图片:
// 参数分别为:请求图片的地址、图片的容器ImageView
ImageView imgView = (ImageView) findViewById(R.id.imgView);
new ImageLoaderUtil().setImageRequest("http://7xinb0.com1.z0.glb.clouddn.com/skin/HopeRebuild/dist/images/logo/logo_40.png", imgView);
在布局文件中,定义ImageView:
<ImageView
android:id="@+id/imgView"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
5.4 通过ImageLoader来显示网络图片:
// 参数分别为:请求图片的地址、图片的容器ImageView、默认显示的图片ResourceID、请求失败时显示的图片的ResourceID
new ImageLoaderUtil().setImageLoader("http://7xinb0.com1.z0.glb.clouddn.com/skin/HopeRebuild/dist/images/logo/logo_40.png", imgView, R.mipmap.default, R.mipmap.error);
在布局文件中,定义ImageView:
<ImageView
android:id="@+id/imgView"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
5.5 通过Volley的NetWorkImageView来显示网络图片:
// 参数分别为:请求图片的地址、图片的容器NetworkImageView、默认显示的图片ResourceID、请求失败时显示的图片的ResourceID
NetworkImageView netWorkImageView = (NetworkImageView) findViewById(R.id.imgNetworkView);
new ImageLoaderUtil().setNetWorkImageView("http://7xinb0.com1.z0.glb.clouddn.com/skin/HopeRebuild/dist/images/logo/logo_40.png", netWorkImageView, R.mipmap.default, R.mipmap.error);
在布局文件中,定义NetworkImageView:
<com.android.volley.toolbox.NetworkImageView
android:id="@+id/imgNetworkView"
android:layout_width="300dp"
android:layout_height="300dp"
android:layout_centerHorizontal="true"/>
六. 后记
该Volley的封装中,暂未考虑到图片和数据缓存。
有一些地方封装得仍不够抽象,有待完善。
非常欢迎读者能提出修改建议,一起进步。
本文作者:凳子_Joinery,转载请注明出处:http://www.dengzhr.com/others/mobile/android/762