android-async-http
retrofit
okhttp
他们各有优劣,之前个人则比较喜欢用android-async-http, 如今Google推出了官方的针对Android平台上的网络通信库,能使网络通信更快,更简单,更健壮,Volley在提供了高性能网络通讯功能的同时,对网络图片加载也提供了良好的支持,完全可以满足简单REST客户端的需求, 我们没有理由不跟上时代的潮流
下载Volley源码并build jar包。
$ git clone https://android.googlesource.com/platform/frameworks/volley $ cd volley $ android update project -p $ ant jar
然后把生成的jar包引用到我们的项目中,extras目录下则包含了目前最新的volley源码。
此Demo主要介绍了日常网络开发常用的基本功能,但volley的扩展性很强,可以根据需要定制你自己的网络请求。
volley视频地址: http://www.youtube.com/watch?v=yhv8l9F44qo&feature=player_embedded
以上是在Google IO的演讲上ppt的配图,从上面这张图我们可以看出,volley适合快速,简单的请求(Json对象,图片加载)。
volley的特性:
接下来,我们来学习简单的使用下volley给我提供的API吧。
1.首先拿到一个请求队列(RequestQueue只需要一个实例即可,不像AsyncTask每次使用都要new一个)
// 初始化RequestQueue一个activity只需要一个 private void initRequestQueue() { mQueue = Volley.newRequestQueue(getApplicationContext()); }
由于用法都相差不大,我就不一一举例了,举几个常用有代表性的例子:
以下代码是StringRequest的get请求:
// get请求
private void loadGetStr(String url) { StringRequest srReq = new StringRequest(Request.Method.GET, url, new StrListener(), new StrErrListener()) { protected final String TYPE_UTF8_CHARSET = "charset=UTF-8"; // 重写parseNetworkResponse方法改变返回头参数解决乱码问题 // 主要是看服务器编码,如果服务器编码不是UTF-8的话那么就需要自己转换,反之则不需要 @Override protected Response<String> parseNetworkResponse( NetworkResponse response) { try { String type = response.headers.get(HTTP.CONTENT_TYPE); if (type == null) { type = TYPE_UTF8_CHARSET; response.headers.put(HTTP.CONTENT_TYPE, type); } else if (!type.contains("UTF-8")) { type += ";" + TYPE_UTF8_CHARSET; response.headers.put(HTTP.CONTENT_TYPE, type); } } catch (Exception e) { } return super.parseNetworkResponse(response); } }; srReq.setShouldCache(true); // 控制是否缓存 startVolley(srReq); }
// post请求 private void loadPostJson(String url) { // 第二个参数说明: // Constructor which defaults to GET if jsonRequest is null, POST // otherwise. // 默认情况下设成null为get方法,否则为post方法。 JsonObjectRequest srReq = new JsonObjectRequest(url, null, new JsonListener(), new StrErrListener()) { @Override protected Map<String, String> getParams() throws AuthFailureError { Map<String, String> map = new HashMap<String, String>(); map.put("w", "2459115"); map.put("u", "f"); return map; } }; srReq.setShouldCache(false); // 控制是否缓存 startVolley(srReq); }
// Str请求成功回调 private class StrListener implements Listener<String> { @Override public void onResponse(String arg0) { Log.e(Tag, arg0); } } // Gson请求成功回调 private class GsonListener implements Listener<ErrorRsp> { @Override public void onResponse(ErrorRsp arg0) { Toast.makeText(mContext, arg0.toString(), Toast.LENGTH_LONG).show(); } } // 共用失败回调 private class StrErrListener implements ErrorListener { @Override public void onErrorResponse(VolleyError arg0) { Toast.makeText(mContext, VolleyErrorHelper.getMessage(arg0, mContext), Toast.LENGTH_LONG).show(); } }接下来是 ImageRequest
/** * 第三第四个参数分别用于指定允许图片最大的宽度和高度,如果指定的网络图片的宽度或高度大于这里的最大值,则会对图片进行压缩, * 指定成0的话就表示不管图片有多大,都不会进行压缩。 * * @param url * 图片地址 * @param listener * @param maxWidth * 指定允许图片最大的宽度 * @param maxHeight * 指定允许图片最大的高度 * @param decodeConfig * 指定图片的颜色属性,Bitmap.Config下的几个常量. * @param errorListener */ private void getImageRequest(final ImageView iv, String url) { ImageRequest imReq = new ImageRequest(url, new Listener<Bitmap>() { @Override public void onResponse(Bitmap arg0) { iv.setImageBitmap(arg0); } }, 60, 60, Bitmap.Config.ARGB_8888, new StrErrListener()); startVolley(imReq); }看到现在大家肯定会疑惑写了这么多不同类型的Request到底如何运行?接下请看:
// 添加及开始请求 private void startVolley(Request req) { // 设置超时时间 // req.setRetryPolicy(new DefaultRetryPolicy(20 * 1000, 1, 1.0f)); // 将请求加入队列 mQueue.add(req); // 开始发起请求 mQueue.start(); }
volley不仅提供了这些请求的方式,还提供了加载图片的一些方法和控件:
比如我们一个列表需要加载很多图片我们可以使用volley给我们提供的ImageLoader( ImageLoader比ImageRequest更加高效,因为它不仅对图片进行缓存,还可以过滤掉重复的链接,避免重复发送请求。)
public class ImageAdapter extends ArrayAdapter<String> { private RequestQueue mQueue; private ImageLoader mImageLoader; public ImageAdapter(Context context, List<String> objects) { super(context, 0, objects); mQueue = Volley.newRequestQueue(getContext()); mImageLoader = new ImageLoader(mQueue, new BitmapCache()); } @Override public View getView(int position, View convertView, ViewGroup parent) { String url = getItem(position); ImageView imageView; if (convertView == null) { imageView = new ImageView(getContext()); } else { imageView = (ImageView) convertView; } // getImageListener(imageView控件对象,默认图片地址,失败图片地址); ImageListener listener = ImageLoader.getImageListener(imageView, android.R.drawable.ic_menu_rotate, android.R.drawable.ic_delete); // get(图片地址,listener,宽,高);自动帮你处理图片的宽高再也不怕大图片的oom了 mImageLoader.get(url, listener,100,200); return imageView; } }当然还需要重写 ImageCache这个类 //使用LruCache再也不用怕加载多张图片oom了
public class <span style="font-family: Arial;">BitmapCache</span><span style="font-family: Arial;"> extends LruCache<String, Bitmap> implements ImageCache {</span> // LruCache 原理:Cache保存一个强引用来限制内容数量,每当Item被访问的时候,此Item就会移动到队列的头部。 当cache已满的时候加入新的item时,在队列尾部的item会被回收。 // 解释:当超出指定内存值则移除最近最少用的图片内存 public static int getDefaultLruCacheSize() { // 拿到最大内存 final int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024); // 拿到内存的八分之一来做图片内存缓存 final int cacheSize = maxMemory / 8; return cacheSize; } public BitmapLruCache() { this(getDefaultLruCacheSize()); } public BitmapLruCache(int sizeInKiloBytes) { super(sizeInKiloBytes); } @Override protected int sizeOf(String key, Bitmap value) { return value.getRowBytes() * value.getHeight() / 1024; } @Override public Bitmap getBitmap(String url) { return get(url); } @Override public void putBitmap(String url, Bitmap bitmap) { put(url, bitmap); } }
<com.android.volley.toolbox.NetworkImageView android:id="@+id/network_image_view" android:layout_width="100dp" android:layout_height="100dp" />使用方法:
private void networkImageViewUse(NetworkImageView iv, String url) { ImageLoader imLoader = new ImageLoader(mQueue, new BitmapLruCache()); iv.setDefaultImageResId(R.drawable.ic_launcher); iv.setErrorImageResId(R.drawable.ic_launcher); iv.setImageUrl(url, imLoader); }
1.activity自动销毁时它会自定取消所有请求。
2.给请求设置标签:
request.setTag("My Tag");
取消所有指定标记的请求:
request.cancelAll("My Tag");Volley的架构设计:
其中蓝色部分代表主线程,绿色部分代表缓存线程,橙色部分代表网络线程。我们在主线程中调用RequestQueue的add()方法来添加一条网络请求,这条请求会先被加入到缓存队列当中,如果发现可以找到相应的缓存结果就直接读取缓存并解析,然后回调给主线程。如果在缓存中没有找到结果,则将这条请求加入到网络请求队列中,然后处理发送HTTP请求,解析响应结果,写入缓存,并回调主线程。接下来还有ym—— Android网络框架Volley(实战篇)