相关文章
Android网络编程(一)HTTP协议原理
Android网络编程(二)HttpClient与HttpURLConnectionhtml
Volley想必许多人都用过,为了创建网络编程的知识体系。Volley是必须要讲的知识点,因此我这里有必要再次介绍一下Volley的使用。java
在2013年Google I/O大会上推出了一个新的网络通讯框架Volley。android
Volley既可以訪问网络取得数据,也可以载入图片,并且在性能方面也进行了大幅度的调整。它的设计目标就是很适合去进行数据量不大。但通讯频繁的网络操做。而对于大数据量的网络操做,比方说下载文件等,Volley的表现就会很糟糕。在使用Volley前请下载Volley库并放在libs文件夹下并add到project中。git
下载Volley请点击这github
Volley请求网络都是基于请求队列的,开发人员仅仅要把请求放在请求队列中就可以了,请求队列会依次进行请求,普通状况下。一个应用程序假设网络请求没有特别频繁则全然可以仅仅有一个请求队列(相应Application)。假设许多或其它状况,则可以是一个Activity相应一个网络请求队列,这就要看详细状况了,首先建立队列:编程
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
StringRequest返回的数据是String类型的,咱们查看下StringRequest的源代码:json
public class StringRequest extends Request<String> {
private final Listener<String> mListener;
public StringRequest(int method, String url, Listener<String> listener, ErrorListener errorListener) {
super(method, url, errorListener);
this.mListener = listener;
}
public StringRequest(String url, Listener<String> listener, ErrorListener errorListener) {
this(0, url, listener, errorListener);
}
...省略
}
有两个构造函数,当中第一个比第二个多了一个请求的方法,假设採用第二个则默认是GET请求。好了。咱们试着用GET方法来请求百度:api
//建立请求队列
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
StringRequest mStringRequest = new StringRequest(Request.Method.GET, "http://www.baidu.com",
new Response.Listener<String>() {
@Override
public void onResponse(String response) {
Log.i("wangshu", response);
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
Log.e("wangshu", error.getMessage(), error);
}
});
//将请求加入在请求队列中
mQueue.add(mStringRequest);
固然别忘了加入网络訪问权限:缓存
<uses-permission android:name="android.permission.INTERNET"/>
请求结果不用说是百度界面的html文件:markdown
和StringRequest类似,咱们直接上代码:
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
JsonObjectRequest mJsonObjectRequest = new JsonObjectRequest(Request.Method.POST,"http://api.1-blog.com/biz/bizserver/article/list.do",
new Response.Listener<JSONObject>() {
@Override
public void onResponse(JSONObject response) {
Log.d("wangshu", response.toString());
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
Log.e("wangshu", error.getMessage(), error);
}
}
);
mQueue.add(mJsonObjectRequest);
执行程序返回的是一堆新闻的Json数据:
为了解析这些Json数据。咱们用Gson来解析Json数据。点击这里下载Gson将jar包放在libs文件夹下并add进project中。咱们開始写article类用于存储数据:
public class Article {
private String desc;
private String status;
private List<detail> detail = new ArrayList<detail>();
public List<Article.detail> getDetail() {
return detail;
}
public void setDetail(List<Article.detail> detail) {
this.detail = detail;
}
public String getDesc() {
return desc;
}
public void setDesc(String desc) {
this.desc = desc;
}
public String getStatus() {
return status;
}
public void setStatus(String status) {
this.status = status;
}
public class detail {
private String title;
private String article_url;
private String my_abstract;
private String article_type;
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getArticle_url() {
return article_url;
}
public void setArticle_url(String article_url) {
this.article_url = article_url;
}
public String getMy_abstract() {
return my_abstract;
}
public void setMy_abstract(String my_abstract) {
this.my_abstract = my_abstract;
}
public String getArticle_type() {
return article_type;
}
public void setArticle_type(String article_type) {
this.article_type = article_type;
}
}
}
最后咱们改写JsonRequest的请求回调:
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
JsonObjectRequest mJsonObjectRequest = new JsonObjectRequest(Request.Method.POST,"http://api.1-blog.com/biz/bizserver/article/list.do",
new Response.Listener<JSONObject>() {
@Override
public void onResponse(JSONObject response) {
Article mArticle=new Gson().fromJson(response.toString(), Article.class);
List<Article.detail>mList=mArticle.getDetail();
String title=mList.get(0).getTitle();
Log.d("wangshu", title);
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
Log.e("wangshu", error.getMessage(), error);
}
}
);
mQueue.add(mJsonObjectRequest);
来看看打印结果:
ImageRequest已是过期的方法了,和前面两种的使用方法类似:
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
ImageRequest imageRequest = new ImageRequest(
"http://img.my.csdn.net/uploads/201603/26/1458988468_5804.jpg",
new Response.Listener<Bitmap>() {
@Override
public void onResponse(Bitmap response) {
iv_image.setImageBitmap(response);
}
}, 0, 0, Bitmap.Config.RGB_565, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
iv_image.setImageResource(R.drawable.ico_default);
}
});
mQueue.add(imageRequest);
查看ImageRequest的源代码发现它可以设置你想要的图片的最大宽度和高度。在载入图片时假设图片超过时望的最大宽度和高度则会进行压缩:
public ImageRequest(String url, Listener<Bitmap> listener, int maxWidth, int maxHeight, ScaleType scaleType, Config decodeConfig, ErrorListener errorListener) {
super(0, url, errorListener);
this.setRetryPolicy(new DefaultRetryPolicy(1000, 2, 2.0F));
this.mListener = listener;
this.mDecodeConfig = decodeConfig;
this.mMaxWidth = maxWidth;
this.mMaxHeight = maxHeight;
this.mScaleType = scaleType;
}
ImageLoader的内部使用ImageRequest来实现,它的构造器可以传入一个ImageCache缓存形參,实现了图片缓存的功能,同一时候还可以过滤反复连接,避免反复发送请求。
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
ImageLoader imageLoader = new ImageLoader(mQueue, new BitmapCache());
ImageLoader.ImageListener listener = ImageLoader.getImageListener(iv_image,R.drawable.ico_default, R.drawable.ico_default);
imageLoader.get("http://img.my.csdn.net/uploads/201603/26/1458988468_5804.jpg", listener);
与ImageRequest实现效果不一样的是。ImageLoader载入图片会先显示默认的图片,等待图片载入完毕才会显示在ImageView上。
固然ImageLoader也提供了设置最大宽度和高度的方法:
public ImageLoader.ImageContainer get(String requestUrl, ImageLoader.ImageListener imageListener, int maxWidth, int maxHeight) {
return this.get(requestUrl, imageListener, maxWidth, maxHeight, ScaleType.CENTER_INSIDE);
}
NetworkImageView是一个本身定义控件,继承自ImageView,封装了请求网络载入图片的功能。
先在布局中引用:
<com.android.volley.toolbox.NetworkImageView
android:id="@+id/nv_image"
android:layout_width="200dp"
android:layout_height="200dp"
android:layout_centerHorizontal="true"
android:layout_below="@id/iv_image"
android:layout_marginTop="20dp"
></com.android.volley.toolbox.NetworkImageView>
代码中调用,和ImageLoader使用方法类似:
iv_image = (ImageView) this.findViewById(R.id.iv_image);
RequestQueue mQueue = Volley.newRequestQueue(getApplicationContext());
ImageLoader imageLoader = new ImageLoader(mQueue, new BitmapCache());
nv_image.setDefaultImageResId(R.drawable.ico_default);
nv_image.setErrorImageResId(R.drawable.ico_default);
nv_image.setImageUrl("http://img.my.csdn.net/uploads/201603/26/1458988468_5804.jpg",
imageLoader);
NetworkImageView并无提供设置最大宽度和高度的方法,依据咱们设置控件的宽和高结合网络图片的宽和高内部会本身主动去实现压缩。假设咱们不想要压缩可以设置NetworkImageView控件的宽和高都为wrap_content。