json数据解析及网络请求

前文已经介绍了ListView和RecyclerView等各种列表控件,在实际的开发中,会经常遇到大批量的数据加载和展示的问题,本节课来具体的学习一下常见的方式和步骤。

一、json解析

移动端和服务器交互一般用得较多的数据传递方式都是 Json 字符串的形式, 保存对象,我们也可以写成一个 Json 字符串然后存储。

常见的解析 Json的方式有Android自带Json解析器, Gson,Fastjson,jackson 等。

1.0 http://Json.cn工具

通过一个在线网站工具可以实时的实现json数据的格式化:http://www.json.cn/

1.1 Android自带的Json解析器

自带的解析器的API都存在于org.json包下,用到的类有下面这些:

  • JSONObject: Json对象,可以完成Json字符串与Java对象的相互转换
  • JSONArray: Json数组,可以完成Json字符串与Java集合或对象的相互转换,[]
  • JSONStringer: Json文本构建类,这个类可以帮助快速和便捷的创建JSON text, 每个JSONStringer实体只能对应创建一个JSON text
  • JSONTokener:Json解析类
  • JSONException:Json异常

使用方法参见案例17的原生解析方法,如下:

private List<Lesson> parseJsonByNative(String jsonStr) {
  List<Lesson> lessons = new ArrayList<>();
  try {
    JSONObject jsonObject = new JSONObject(jsonStr);
    int code = jsonObject.getInt("status");
    if (code != 1) {
      return null;
    }
    JSONArray dataArray = jsonObject.getJSONArray("data");
    for (int i = 0; i < dataArray.length(); i++) {
      JSONObject lessonObject = dataArray.getJSONObject(i);
      int id = lessonObject.getInt("id");
      String name = lessonObject.getString("name");
      String picSmall = lessonObject.getString("picSmall");
      String picBig = lessonObject.getString("picBig");
      String description = lessonObject.getString("description");
      int learner = lessonObject.getInt("learner");
​
      Lesson lesson = new Lesson();
      lesson.setId(id);
      lesson.setName(name);
      lesson.setPicSmall(picSmall);
      lesson.setPicBig(picBig);
      lesson.setDescription(description);
      lesson.setLearner(learner);
​
      lessons.add(lesson);
    }
  } catch (JSONException e) {
    e.printStackTrace();
  }
  return lessons;
}

1.2 Gson解析库

Gson是google开源的一款用于json解析的库,受到很多开发者的喜爱。

Gson在github的开源库地址:https://github.com/google/gson

在Android中使用gson有两种方式:

  • 在build.gradle文件中设置库依赖:implementation 'com.google.code.gson:gson:2.8.6'
  • 下载gson.jar包,放在Android项目恶libs目录中

设置依赖后,即可在项目中使用Gson解析数据,如下所示:

private Lesson[] parseByGson(String jsonStr) {
  Gson gson = new Gson();
  Type type = new TypeToken<Common<Lesson>>() {
  }.getType();
  Common<Lesson> common = gson.fromJson(jsonStr, type);
  return common.data;
}

1.3 FastJson解析库

fastjson是阿里巴巴团队开源的一款json解析库,使用的项目和团队也很多。在github上的star数达到了21.2k,超过了gson。

FastJson在复杂类型的Bean转换Json上会出现一些问题,可能会出现引用的类型,导致Json转换出错,需要制定引用。FastJson采用独创的算法,将parse的速度提升到极致,超过所有json库。

fastjson开源库在github的开源地址:https://github.com/alibaba/fastjson

在Android中使用fastjson,需要在项目的build.gradle文件中设置依赖:

compile 'com.alibaba:fastjson:1.1.71.android'

同样的和gson一样,也可以下载最新的jar文件,放在android项目的libs目录中进行依赖设置。

public static List<Lesson> parseByFastJson(String jsonStr) {
  com.alibaba.fastjson.JSONObject jsonObject = JSON.parseObject(jsonStr);
  com.alibaba.fastjson.JSONArray jsonArray = jsonObject.getJSONArray("data");
  return  jsonArray.toJavaList(Lesson.class);
}

1.4 Jackson

Jackson是当前用的比较广泛的,用来序列化和反序列化json的Java开源框架。Jackson是最流行的json解析器之一,Java应用框架中很多都是使用Jackson库。目前github上Jackson的star数量达到5.5K。

Jackson在github上的地址如下:https://github.com/FasterXML/jackson

使用Jackson:

  • 先下载jar包
  • 放入到项目的libs目录中,设置依赖为library
  • 在x项目中进行使用

此处省略案例,可自行练习。

1.5 解析库对比和总结

  • 反序列化操作:总体上JSONObject处理性能比较突出,但是JSONObject的缺点是啰嗦,代码量多,需要些try...catch等冗余代码。
  • Gson:随着数据量的增大,Gson库耗时会增加。
  • FastJson操作:性能比较好,数据的量级对内存消耗影响较小,且消耗内存比较小。
  • JSONObject原生:数据量大时,JSONObject内存消耗很明显。
  • Jackson:耗时和内存消耗上都较大。

① 数据量小时,可以选择JSONObject进行处理,要接收代码量冗余。Gson和FastJson差不多。

② 数据量大时,或者数据量会有明显的量级的增加变化,FastJson综合表现更好。

二、加载网络数据的注意事项

json数据通常都是通过网络接口请求而来,因此需要进行网络访问获取数据,包括图片的加载也是需要网络的。

uses-permission网络权限

使用网络,需要在AndroidMainfest.xml文件中设置使用网络权限。

<!--网络权限-->
<uses-permission android:name="android.permission.INTERNET"/>

Https规范

在Android P版本开始,为了安全起见,不在允许使用http形式的接口。为了解决这个问题,可以通过在AndroidManifest.xml文件的Application文件中,设施usesCleartextTraffic属性为true来,表示使用明文网络流量。推荐还是尽快转换接口升级为https。

数据请求更新规范

① 在Android中,不能在主线程请求网络,否则会报错,网络请求需要放在工作子线程中。

② 在Android中,请求数据完毕,不能在子线程中操作UI,否则会报错。

图片加载开源库

图片加载在android中可以使用比较成熟的开源库来解决,常用的图片加载开源库是Glide,picasso等。

glide

glide在github上的开源地址是:https://github.com/bumptech/glide

使用时首先需要引入glide库的依赖:

//图片网络库
implementation 'com.github.bumptech.glide:glide:4.11.0'
annotationProcessor 'com.github.bumptech.glide:compiler:4.11.0'

在图片控件加载的地方,执行如下操作:

 Glide.with(mContext).load(lesson.getPicBig()).into(holder.imgCover);

picasso

picasso在github上的开源地址是:https://github.com/square/picasso

使用时首先需要引入glide库的依赖:

implementation 'com.squareup.picasso:picasso:2.71828'

在图片控件加载的地方,执行如下操作:

Picasso.get().load("http://i.imgur.com/DvpvklR.png").into(imageView);

三、Android常见的网络请求

3.1 HttpURLConnection

与java的基础相同,HttpURLConnection可以实现网络请求。具备步骤为:

  • 构建URL。
  • openConnection打开HttpURLConnection对象。
  • 通过IO流读取数据。

3.2 Volley库

在Goole I/O 2013上发布的网络通信库,取名为Volley。

Volley的优势及特点是:自动调度网络请求、支持请求优先级、支持取消请求,可以取消单个请求或多个,并且Volley回调时候是在主线程,可以直接操作UI。

Volley库的操作和使用说明文档,可以访问如下地址:https://developer.android.com/training/volley/index.html

设置库依赖

implementation 'com.android.volley:volley:1.1.1'

RequestQueue请求列队

请求之前,首先需要构建RequestQueue请求列队,全局也只初始化一次就好。

Request请求对象

Request请求对象,有StringRequest、ImageRequest、ClearCacheRequest、JsonRequest这四个子类。

可以给Request设置一个tag,并通过RequestQueue.cancelAll(tag)可以进行取消。

3.3 Okhttp

第二种方式是使用成熟的网络请求库,比如说Okhttp,非常火的一个网络库,由Square公司提供。

Okhttp开源库的github地址如下:https://github.com/square/okhttp

最新的Okhttp库的版本是4.6.0,在Android应用中使用该库的操作方法是设置库依赖,在build.gradle文件中:

implementation("com.squareup.okhttp3:okhttp:4.6.0")

3.3.1 使用步骤

  • 构建OkHttpClient对象:全局只需要初始化一次即可。
  • Request对象:通过Builder进行构建和参数设置。
  • 请求:get请求直接使用.get();post请求通过.post(RequestBody),携带数据。
OkHttpClient client = new OkHttpClient();

String run(String url) throws IOException {
  Request request = new Request.Builder()
      .url(url)
      .build();
  try (Response response = client.newCall(request).execute()) {
    return response.body().string();
  }
}

OkHttp库的请求,回调方法是运行在子线程中,因此不能直接在回调函数更新ui操作,如果在子线程中操作UI,刷新界面操作,会报如下错误:

Only the original thread that created a view hierarchy can touch its views

如果出现该错误,可以使用如下的方法解决。该方法用于回到主线程操作UI:

...
final String jsonStr = response.body().string();
runOnUiThread(new Runnable() {
  @Override
  public void run() {
    DialogUtils.closeDialog(mDialog);
    mData = Data.parseByFastJson(jsonStr);
    setDataAndAdapter();
  }
});
...