相關文章
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。