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 ...
随机推荐
- js函数自执行
在javascript里,任何function在执行的时候都会创建一个执行上下文,因为function声明的变量和function有可能只在该function内部,这个上下文,在调用function的 ...
- python怎么装模块
windows下 最简单的方法: File---Settings--Project ---Project Interpreter 下----点击 +号,输入你需要安装的模块名,点击Install P ...
- Android动画的使用总结
1.补间动画(透明渐变.平移.旋转.缩放.组合) 方法一:通过xml文件设置 1-1:创建:res/anim 1-2:java代码写调用 Animation a = AnimationUtils.lo ...
- java poi Excel导入 整数浮点数转换问题解决
/** * 获取单元格数据 */ protected static String getCellValue(Cell cell) { String cellValu ...
- Javascript中setTimeout()的用法详解
1.SetTimeOut() 1.1 SetTimeOut()语法例子 1.2 用SetTimeOut()执行Function 1.3 SetTimeout()语法 ...
- NSIS学习记录の----NSIS插件调用
我们都知道NSIS可以和C或者C++混合编程,方法是NSIS调用C或C++的动态库,那么如何调用呢? 首先我们来创建动态库: // add.cpp : 定义 DLL 应用程序的导出函数. #inclu ...
- sql 百万级数据库优化方案
转自http://blog.sina.com.cn/s/blog_724cd89d0100ppcz.html 1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉 ...
- 华为S9306简单实用配置合集
华为QuidWay交换机配置命令手册: .开始 建立本地配置环境,将主机的串口通过配置电缆与以太网交换机的Console口连接. 在主机上运行终端仿真程序(如Windows的超级终端等),设置终端通信 ...
- Android编译选项eng、user、userdebug的区别
eng:debug 版本 user: release 版本 userDebug版本:部分debug版本 LOCAL_MODULE_TAGS := user eng optional test这个样子. ...
- application/x-www-form-urlencoded
在Form元素的语法中,EncType表明提交数据的格式 用 Enctype 属性指定将数据回发到服务器时浏览器使用的编码类型. 下边是说明: application/x-www-form-urlen ...