Android Volley和Gson实现网络数据加载
Android Volley和Gson实现网络数据加载
先看接口
1 升级接口
http://s.meibeike.com/mcloud/ota/cloudService
POST请求
参数列表如下
mainversion 1.0 commoninfo {"timestamp":"1450772874213","clientversion":"6.0","clientdescription":"goof","commonversion":"1.0","format":"json","clienttype":"3"} json {"yunbangsn":"","pcode":"ncc","subversion":0,"clienttype":"3","innerversion":2015121821,"function":118}
返回信息如下
{
"code": 0,
"type": "0",
"url": "http://d.meibeike.com/mcfs/ota/client/2015-12-18/CloudBar_2015121818_1.0.2.9.apk",
"function": 118,
"message": "成功获取版本信息",
"displayversion": "1.0.2.9",
"filesize": 6363546,
"subversion": 0,
"innerversion": 2015121818,
"updatemsg": "1.修改投射提示“USB导入出错”问题\r\n2.修改偶然性崩溃\r\n3.修改偶然性已绑定用户由于某些操作后变成了未绑定用户,数据丢失",
"updatemsg_cn": "1.修改投射提示“USB导入出错”问题\r\n2.修改偶然性崩溃\r\n3.修改偶然性已绑定用户由于某些操作后变成了未绑定用户,数据丢失",
"md5": "727c136c7914ffa8f480bf4d2037871d",
"updatemsg_tw": null,
"forceflag": "0"
}
上代码
public class MainActivity extends Activity { TextView show; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); show = (TextView)findViewById(R.id.show); Button test = (Button)findViewById(R.id.test);
test.setOnClickListener(new View.OnClickListener() { @Override
public void onClick(View v) { RequestQueue queue = MyVolley.getRequestQueue();
GsonRequestNew<Cloud103Entity> myReq = new GsonRequestNew<Cloud103Entity>(Method.POST,
"http://192.168.2.33:8084/mcloud/cloudclub/cloudService",
Cloud103Entity.class,
createMyReqSuccessListener(),
createMyReqErrorListener()){
//传参数
protected Map<String, String> getParams() throws com.android.volley.AuthFailureError {
Map<String, String> params = new HashMap<String, String>();
params.put("mainversion", "1.0");
params.put("commoninfo", getCommoninfo());
params.put("json", getJson());
Log.i("TAG", "发送参数" + params);
return params;
}; }; Log.i("TAG", "发送URL" + "http://192.168.2.33:8084/mcloud/cloudclub/cloudService"); queue.add(myReq); }
});
} protected String getCommoninfo() { JSONObject commoninfo = new JSONObject();
try {
commoninfo.put("commonversion", "1.0");
commoninfo.put("clienttype", "3");
commoninfo.put("clientversion", "6.0");
commoninfo.put("clientdescription", "goof");
commoninfo.put("timestamp", "1450765392640");
commoninfo.put("format", "json");
} catch (JSONException e) {
Log.i("TAG", "e---" + e.getMessage());
} return commoninfo.toString();
} protected String getJson() { JSONObject json = new JSONObject();
try {
json.put("function", 103);
json.put("subversion", 0);
json.put("userid", "mm730@mbk.com");
json.put("password", "e10adc3949ba59abbe56e057f20f883e");
json.put("accounttype", 0);
} catch (JSONException e) {
Log.i("TAG", "e---" + e.getMessage());
} return json.toString();
} private Response.Listener<Cloud103Entity> createMyReqSuccessListener() {
return new Response.Listener<Cloud103Entity>() {
@Override
public void onResponse(Cloud103Entity response) {
show.setText(response.getMessage());
}
};
} private Response.ErrorListener createMyReqErrorListener() {
return new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
show.setText(error.getMessage());
}
};
} }
public class GsonRequestNew<T> extends Request<T> {
private final Listener<T> mListener; private Gson mGson; private Class<T> mClass; public GsonRequestNew(int method, String url, Class<T> clazz, Listener<T> listener, ErrorListener errorListener) {
super(method, url, errorListener);
mGson = new Gson();
mClass = clazz;
mListener = listener;
} public GsonRequestNew(String url, Class<T> clazz, Listener<T> listener, ErrorListener errorListener) {
this(Method.GET, url, clazz, listener, errorListener);
} @Override
protected Response<T> parseNetworkResponse(NetworkResponse response) {
try {
String jsonString = new String(response.data, HttpHeaderParser.parseCharset(response.headers));
Log.i("TAG", "响应" + jsonString);
return Response.success(mGson.fromJson(jsonString, mClass), HttpHeaderParser.parseCacheHeaders(response));
} catch (UnsupportedEncodingException e) {
return Response.error(new ParseError(e));
}
} @Override
protected void deliverResponse(T response) {
mListener.onResponse(response);
} }
public class Cloud103Entity { private ArrayList<String> bindlist;
private String message;// 返回信息
private String sessionkey;// 登录成功后,后续操作以此sessionKey为准
private String username;
private short subversion = 0;// 业务接口协议版本号
private String email;// email号码账号
private long meiid;// 美贝壳内部id
private int faceid;// 用户头像id
private int code;// 返回码
private String faceurl;// 用户头像url
private String mobile;// 手机号码账号
private int function; public ArrayList<String> getBindlist() {
return this.bindlist;
} public void setBindlist(ArrayList<String> bindlist) {
this.bindlist = bindlist;
} public String getMessage() {
return this.message;
} public void setMessage(String message) {
this.message = message;
} public String getSessionkey() {
return this.sessionkey;
} public void setSessionkey(String sessionkey) {
this.sessionkey = sessionkey;
} public String getUsername() {
return this.username;
} public void setUsername(String username) {
this.username = username;
} public short getSubversion() {
return this.subversion;
} public void setSubversion(short subversion) {
this.subversion = subversion;
} public String getEmail() {
return this.email;
} public void setEmail(String email) {
this.email = email;
} public long getMeiid() {
return this.meiid;
} public void setMeiid(long meiid) {
this.meiid = meiid;
} public int getFaceid() {
return this.faceid;
} public void setFaceid(int faceid) {
this.faceid = faceid;
} public int getCode() {
return this.code;
} public void setCode(int code) {
this.code = code;
} public String getFaceurl() {
return this.faceurl;
} public void setFaceurl(String faceurl) {
this.faceurl = faceurl;
} public String getMobile() {
return this.mobile;
} public void setMobile(String mobile) {
this.mobile = mobile;
} public int getFunction() {
return this.function;
} public void setFunction(int function) {
this.function = function;
} /**
*
* { "bindlist": [ "C001B00010000008" ], "message": "登录成功", "sessionkey":
* "DC385A65D2FEA3FB2021F3F717B4FCD1", "username": "", "subversion": 0,
* "email": "mm730@mbk.com", "meiid": 1000023, "faceid": 0, "code": 0,
* "faceurl": "", "mobile": "", "function": "103" }
*/
}
public class MyApp extends Application{ @Override
public void onCreate() {
super.onCreate(); MyVolley.init(this);
} }
public class MyVolley { private static RequestQueue mRequestQueue; static void init(Context context) {
mRequestQueue = Volley.newRequestQueue(context);
} public static RequestQueue getRequestQueue() {
if (mRequestQueue != null) {
return mRequestQueue;
} else {
throw new IllegalStateException("RequestQueue not initialized");
}
}
}
Code见 https://github.com/huanyi0723/VolleyGson/
Volley代码剖析
Volley
和 Volley 框架同名的类,其实是个工具类,作用是构建一个可用于添加网络请求的 RequestQueue 对象
Request
网络请求的抽象类。我们通过构建一个 Request 类的非抽象子类(StringRequest、JsonRequest、ImageRequest 或自定义)对象
Volley 支持 8 种 Http 请求方式 GET, POST, PUT, DELETE, HEAD, OPTIONS, TRACE, PATCH
Request 类中包含了请求 url,请求请求方式,请求 Header,请求 Body,请求的优先级等信息
子类必须重写的两个方法
parseNetworkResponse() //将网络返回的原生字节内容,转换成合适的类型
deliverResponse() //将解析成合适类型的内容传递给它们的监听回调
getBody() //可以构建用于 POST、PUT、PATCH 请求方式的 Body 内容
getParams() // getBody 函数没有被重写情况下,此方法的返回值会被 key、value 分别编码后拼装起来转换为字节码作为 Body 内容
RequestQueue
Volley 框架的核心类,将请求 Request 加入到一个运行的 RequestQueue 中,来完成请求操作
mCacheQueue //缓存请求队列
mNetworkQueue //网络请求队列
mCurrentRequests //正在进行中,尚未完成的请求集合
mWaitingRequests //等待请求的集合
start() //开启一个缓存调度线程 CacheDispatcher 和 n 个网络调度线程 NetworkDispatcher ,这里 n 默认为 4
add() //加入请求
finish() //请求完成
cancelAll() //请求取消
CacheDispatcher
一个线程,用于调度处理走缓存的请求
mCacheQueue //缓存请求队列
mNetworkQueue //网络请求队列
mCache //代表了一个可以获取请求结果,存储请求结果的缓存
mDelivery //请求结果传递类
NetworkDispatcher
一个线程,用于调度处理走网络的请求
mQueue //网络请求队列
mNetwork //网络类,代表了一个可以执行请求的网络
mCache //缓存类
mDelivery //请求结果传递类
Cache
缓存接口,代表了一个可以获取请求结果,存储请求结果的缓存
get() //通过 key 获取请求的缓存实体
put() //存入一个请求的缓存实体
remove() //移除指定的缓存实体
clear() //清空缓存
内部类 Entry //代表缓存实体
data //请求返回的数据(Body 实体)
etag //Http 响应首部中用于缓存新鲜度验证的 ETag
serverDate //Http 响应首部中的响应产生时间
ttl //缓存的过期时间
softTtl //缓存的新鲜时间
responseHeaders //响应的 Headers
isExpired() //判断缓存是否过期,过期缓存不能继续使用
refreshNeeded() //判断缓存是否新鲜,不新鲜的缓存需要发到服务端做新鲜度的检测
DiskBasedCache
继承 Cache 类,基于 Disk 的缓存实现类
initialize() //初始化,扫描缓存目录得到所有缓存数据摘要信息放入内存
get() //从缓存中得到数据。先从摘要信息中得到摘要信息,然后读取缓存数据文件得到内容
put() //将数据存入缓存内。先检查缓存是否会满,会则先删除缓存中部分数据,然后再新建缓存文件
pruneIfNeeded() //检查是否能再分配 neededSpace 字节的空间,如果不能则删除缓存中部分数据
remove() //移除指定的缓存实体
clear() //清空缓存
内部类CacheHeader //缓存文件摘要信息,存储在缓存文件的头部
NoCache
继承 Cache 类,不做任何操作的缓存实现类,可将它作为构建 RequestQueue 的参数以实现一个不带缓存的请求队列
Network
代表网络的接口,处理网络请求
performRequest() //用于执行特定请求
NetworkResponse
Network 中方法 performRequest 的返回值
封装了网络请求响应的 StatusCode,Headers 和 Body 等
statusCode //Http 响应状态码
data //Body 数据
headers //表示是否为 304 响应
networkTimeMs //请求耗时
BasicNetwork
Volley 中默认的网络接口实现类。调用 HttpStack 处理请求,
并将结果转换为可被 ResponseDelivery 处理的 NetworkResponse
HttpStack
用于处理 Http 请求,返回请求结果的接口
performRequest() //执行 Request 代表的请求,第二个参数表示发起请求之前,添加额外的请求 Headers
HttpClientStack
实现 HttpStack 接口,利用 Apache 的 HttpClient 进行各种请求方式的请求
HurlStack
实现 HttpStack 接口,利用 Java 的 HttpURLConnection 进行各种请求方式的请求
Response
封装了经过解析后的数据,用于传输。
并且有两个内部接口 Listener 和 ErrorListener 分别可表示请求失败和成功后的回调。
ByteArrayPool
byte[] 的回收池,用于 byte[] 的回收再利用,减少了内存的分配和回收
PoolingByteArrayOutputStream
继承 ByteArrayOutputStream,原始 ByteArrayOutputStream 中用于接受写入 bytes 的 buf,
每次空间不足时便会 new 更大容量的 byte[],
而 PoolingByteArrayOutputStream 使用了 ByteArrayPool 作为 Byte[] 缓存来减少这种操作,从而提高性能
HttpHeaderParser
Http header 的解析工具类,在 Volley 中主要作用是用于解析 Header 从而判断返回结果是否需要缓存,
如果需要返回 Header 中相关信息
parseDateAsEpoch() //解析时间,将 RFC1123 的时间格式,解析成 epoch 时间
parseCharset() //解析编码集,在 Content-Type 首部中获取编码集,如果没有找到,默认返回 ISO-8859-1
parseCacheHeaders() 通过网络响应中的缓存控制 Header 和 Body 内容,构建缓存实体。
如果 Header 的 Cache-Control 字段含有 no-cache 或 no-store 表示不缓存,返回 null
RetryPolicy
重试策略接口
getCurrentTimeout() //获取当前请求用时
getCurrentRetryCount() //获取已经重试的次数
retry() //确定是否重试
DefaultRetryPolicy
Volley 默认的重试策略实现类
mCurrentRetryCount //已经重试次数
mBackoffMultiplier
mCurrentTimeoutMs //当前重试的 timeout 时间
ResponseDelivery
请求结果的传输接口,用于传递请求结果或者请求错误
postResponse() //此方法用于传递请求结果, request 和 response 参数分别表示请求信息和返回结果信息
postResponse() //此方法用于传递请求结果,并在完成传递后执行 Runnable
postError() //传输请求错误
ExecutorDelivery
请求结果传输接口具体实现类
在 Handler 对应线程中传输缓存调度线程或者网络调度线程中产生的请求结果或请求错误,
会在请求成功的情况下调用 Request.deliverResponse(…) 函数,失败时调用 Request.deliverError(…) 函数
Android Volley和Gson实现网络数据加载的更多相关文章
- Android之MVP模式实现登录和网络数据加载
MVP简介 相信大家对 MVC 都是比较熟悉了:M-Model-模型.V-View-视图.C-Controller-控制器,MVP作为MVC的演化版本,也是作为用户界面(用户层)的实现模式,那么类似的 ...
- Android 学习笔记之Volley(八)实现网络图片的数据加载
PS:最后一篇关于Volley框架的博客... 学习内容: 1.使用ImageRequest.java实现网络图片加载 2.使用ImageLoader.java实现网络图片加载 3.使用NetWork ...
- android 网络异步加载数据进度条
ProgressDialog progressDialog = null; public static final int MESSAGETYPE = 0; private void execute( ...
- Android下设置ListView数据加载完成后执行layoutanimation
今天使用android的volley框架写了一个简单的网络天气获取的demo. 承载数据的空间是ListView 因为是网络加载,必然先要设置ListView的默认数据,我设置的就是那个Loading ...
- Google官方网络框架-Volley的使用解析Json以及加载网络图片方法
Google官方网络框架-Volley的使用解析Json以及加载网络图片方法 Volley是什么? Google I/O 大会上,Google 推出 Volley的一个网络框架 Volley适合什么场 ...
- android优化从网络中加载图片速度。。
从网络中加载图片主要要注意两个方面的问题: 1.内存管理:图片占的内存很大,假如图片数量多,很容易让系统抛出out of memory的异常. 同时我们也要注意不同android版本中内存管理的区别. ...
- Android中使用Gson解析JSON数据的两种方法
Json是一种类似于XML的通用数据交换格式,具有比XML更高的传输效率;本文将介绍两种方法解析JSON数据,需要的朋友可以参考下 Json是一种类似于XML的通用数据交换格式,具有比XML更高的 ...
- Android Launcher分析和修改4——初始化加载数据
上面一篇文章说了Launcher是如何被启动的,Launcher启动的过程主要是加载界面数据然后显示出来, 界面数据都是系统APP有关的数据,都是从Launcher的数据库读取,下面我们详细分析Lau ...
- Android: 阻止ScrollView随着数据加载自动滚动
当ScrollView中有类似GridView的控件时,当数据加载后ScrollView会自动滚动.要阻止这种事情发生,我们需要做的是在ScrollView的下层容器中添加android:descen ...
随机推荐
- python: indentationerror: unexpected indent
以后遇到了IndentationError: unexpected indent你就要知道python编译器是在告诉你“Hi,老兄,你的文件里格式不对了,可能是tab和空格没对齐的问题,你需要检查下t ...
- 最全面的Java多线程用法解析
1.创建线程 在Java中创建线程有两种方法:使用Thread类和使用Runnable接口.在使用Runnable接口时需要建立一个Thread实例.因此,无论是通过Thread类还是Runnable ...
- java对象equals方法的重写
根类Object中的equals方法描述: public boolean equals(Object obj)The equals method for class Object implements ...
- poj3270 Cow Sorting
给定有序数组a[1...n]的一个置换a[σ(1)...σ(n)], 通过交换数组元素把置换后的数组恢复为有序, 定义进行一次交换的代价为两元素之和,试问此过程的最小总代价. 实际上一种置换即定义S ...
- 使用 /proc 文件系统来访问 linux操作系统 内核的内容 && 虚拟文件系统vfs及proc详解
http://blog.163.com/he_junwei/blog/static/19793764620152743325659/ http://www.01yun.com/other/201304 ...
- linux epoll 学习
一.epoll介绍 epoll是linux内核为处理大批量句柄而作的改进的poll,是linux下IO多路复用select.poll的增强版,它能显著减少程序在大量并发连接中只有少量活跃的情况下的系统 ...
- 附加数据库对于服务器失败(Microsoft.SqlServer.Smo),无法升级数据库,因为它是只读的,或者具有只读文件
今天在将一个 SQL Server 2000 数据库附加到 SQL Server 2005时出现如下的错误:附加数据库对于服务器失败(Microsoft.SqlServer.Smo),无法升级数据库t ...
- HTTP缓存机制
缓存对于移动端是非常重要的存在. 减少请求次数,减小服务器压力. 本地数据读取速度更快,让页面不会空白几百毫秒. 在无网络的情况下提供数据. 缓存一般由服务器控制(通过某些方式可以本地控制缓存,比如向 ...
- 一个Public的字段引起的,谈谈继承中的new
一直觉得对c#面向对象这块已经掌握的很好了,因为正常情况下字段一般我们设计成私有的,今天突然想到一个实验,如下有两个很简单的类: public class Farther { ; public vir ...
- 浅思OC的语言特性
算了算,学习IOS已经有一段时间了.今天花了点时间思考一下OC的语言特性,让自己的心不要那么浮躁,注重基础,回归本源. OC做为一门面向对象语言,自然具有面向对象的语言特性,如封装.继承.多态.他具有 ...