Android中JSON的4种解析方式使用和对比

目录
  • 1 Android SDK自带的org.json解析
  • 2 Gson解析
  • 3 Jackson解析
  • 4 Fastjson解析

1 Android SDK自带的org.json解析

解析原理: 基于文档驱动,需要把全部文件读入到内存中,然后遍历所有数据,根据需要检索想要的数据。

相关类:

  • JSONObject
  • JSONArray
  • JSONTokener
public Object nextValue() throws JSONException {
    int c = nextCleanInternal();
    switch (c) {
        case -1:
            throw syntaxError("End of input");
            
        case '{':
            return readObject();
            
        case '[':
            return readArray();
            
        case ''':
        case '"':
            return nextString((char) c);
            
        default:
            pos--;
            return readLiteral();
    }
}
  • JSONStringer
  • JSONException

以下是Android SDK自带Json解析方式的示例代码:

/**
 * @Description: SDK org.json下自带的Json解析方式
 * @CreateDate: 2022/3/22 10:30 上午
 */
public class OrgJsonUtil {
    /**
     * 生成Json
     */
    public static void createJson(Context context) {
        try {
            File file = new File(context.getFilesDir(), "orgjson.json");
            //实例化一个JSONObject
            JSONObject student = new JSONObject();
            //向对象中添加数据
            student.put("name", "Musk");
            student.put("sex", "男");
            student.put("age", 50);

            JSONObject course1 = new JSONObject();
            course1.put("name", "数学");
            course1.put("score", 98.2f);
            JSONObject course2 = new JSONObject();
            course2.put("name", "语文");
            course2.put("score", 99);

            //实例化一个JSONArray
            JSONArray courses = new JSONArray();
            courses.put(0, course1);
            courses.put(1, course2);
            student.put("courses", courses);

            //写数据
            FileOutputStream fos = new FileOutputStream(file);
            fos.write(student.toString().getBytes());
            fos.close();
            Log.d("TAG", "createJson: " + student);

            Toast.makeText(context, "Json创建成功", Toast.LENGTH_SHORT).show();

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 解析Json
     */
    public static void parseJson(Context context) {
        try {
            //读数据
            File file = new File(context.getFilesDir(), "orgjson.json");
            FileInputStream fis = new FileInputStream(file);
            InputStreamReader isr = new InputStreamReader(fis);
            BufferedReader br = new BufferedReader(isr);
            String line;
            StringBuffer sb = new StringBuffer();
            while (null != (line = br.readLine())) {
                sb.append(line);
            }
            fis.close();
            isr.close();
            br.close();

            Student student = new Student();
            JSONObject studentJsonObject = new JSONObject(sb.toString());

            //获取对象
            //使用optString在获取不到对应值的时候会返回空字符串"",而使用getString会异常
            String name = studentJsonObject.optString("name", "");
            String sex = studentJsonObject.optString("sex", "女");
            int age = studentJsonObject.optInt("age", 18);
            student.setName(name);
            student.setSex(sex);
            student.setAge(age);

            //获取数组
            List<Course> courses = new ArrayList<>();
            JSONArray coursesJsonArray = studentJsonObject.optJSONArray("courses");
            if (coursesJsonArray != null && coursesJsonArray.length() > 0) {
                for (int i = 0; i < coursesJsonArray.length(); i++) {
                    JSONObject courseJsonObject = coursesJsonArray.optJSONObject(i);
                    Course course = new Course();
                    String courseName = courseJsonObject.optString("name", "学科");
                    float score = (float) courseJsonObject.optDouble("score", 0);
                    course.setName(courseName);
                    course.setScore(score);
                    courses.add(course);
                }
            }
            student.setCourses(courses);

            Log.d("TAG", "parseJson: " + student);
            Toast.makeText(context, "Json解析成功", Toast.LENGTH_SHORT).show();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

2 Gson解析

解析原理: 基于事件驱动。

优势:

  • 快速,高效
  • 代码量少
  • 面向对象
  • 数据传输解析方便
  • 可按需解析

解析流程: 根据所需取的数据 建立1个对应于JSON数据的JavaBean类,即可通过简单操作解析出所需数据。
Gson 不要求JavaBean类里面的属性一定全部和JSON数据里的所有key相同,可以按需取数据。

使用:

  • JSON的大括号对应一个对象
    • 对象里面有key,value
    • JavaBean的类属性名 = key
  • JSON的中括号对应一个数组
  • JavaBean里面对应的也是数组
  • 对象里可以有值/对象
  • 若对象里面只有值,没有key,则说明是纯数组,对应JavaBean里的数组类型
  • 若对象里面有值和key,则说明是对象数组,对应JavaBean里的内部类
  • 对象嵌套
    • 建立内部类 该内部类对象的名字 = 父对象的key ,类似对象数组
{
    "key":"value",
    "simpleArray":[
        1,
        2,
        3
    ],
    "arrays":[
        {
            "arrInnerClsKey":"arrInnerClsValue",
            "arrInnerClsKeyNub":1
        }
    ],
    "innerclass":{
        "name":"Musk",
        "age":50,
        "sex":"男"
    }
}

与之对应的JavaBean:

public class JsonJavaBean {
    private String key;
    private List<Integer> simpleArray;
    private List<ArraysBean> arrays;
    private InnerclassBean innerclass;

    public String getKey() {
        return key;
    }

    public void setKey(String key) {
        this.key = key;
    }

    public List<Integer> getSimpleArray() {
        return simpleArray;
    }

    public void setSimpleArray(List<Integer> simpleArray) {
        this.simpleArray = simpleArray;
    }

    public List<ArraysBean> getArrays() {
        return arrays;
    }

    public void setArrays(List<ArraysBean> arrays) {
        this.arrays = arrays;
    }

    public InnerclassBean getInnerclass() {
        return innerclass;
    }

    public void setInnerclass(InnerclassBean innerclass) {
        this.innerclass = innerclass;
    }

    private static class ArraysBean {
        private String arrInnerClsKey;
        private int arrInnerClsKeyNub;

        public String getArrInnerClsKey() {
            return arrInnerClsKey;
        }

        public void setArrInnerClsKey(String arrInnerClsKey) {
            this.arrInnerClsKey = arrInnerClsKey;
        }

        public int getArrInnerClsKeyNub() {
            return arrInnerClsKeyNub;
        }

        public void setArrInnerClsKeyNub(int arrInnerClsKeyNub) {
            this.arrInnerClsKeyNub = arrInnerClsKeyNub;
        }
    }

    private static class InnerclassBean {
        private String name;
        private int age;
        private String sex;

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public int getAge() {
            return age;
        }

        public void setAge(int age) {
            this.age = age;
        }

        public String getSex() {
            return sex;
        }

        public void setSex(String sex) {
            this.sex = sex;
        }
    }
}

Gson解析示例代码:

/**
 * @Description: Gson使用示例
 * @CreateDate: 2022/3/22 12:51 下午
 */
public class GsonUse {

    public static void main(String[] args) throws Exception {
        Student student = new Student();
        student.setName("John");
        student.setAge(20);
        student.setSex("男");
        List<Course> courses = new ArrayList<>();
        courses.add(new Course("语文", 99));
        courses.add(new Course("地理", 59));
        student.setCourses(courses);

        Gson gson = new Gson();

        //在项目根目录下的json文件夹下生成Jgsonjson.json文件
        File file = new File("json/gsonjson.json");
        FileOutputStream fos = new FileOutputStream(file);
        OutputStreamWriter osw = new OutputStreamWriter(fos, "utf-8");
        JsonWriter jw = new JsonWriter(osw);
        gson.toJson(student, new TypeToken<Student>() {
        }.getType(), jw);
        jw.flush();
        jw.close();

        //反序列化为JavaBean
        FileInputStream fis = new FileInputStream(file);
        InputStreamReader isr = new InputStreamReader(fis);
        JsonReader jr = new JsonReader(isr);
        Student student1 = gson.fromJson(jr, new TypeToken<Student>() {
        }.getType());
        System.out.println(student1);
    }
}

3 Jackson解析

解析原理: 基于事件驱动。

优势: 解析效率最高;在数据量大的情况优势尤为明显、占存少。

缺点: Json数据里面的所有key都有所对应,也就是必须完全解析文档,如果要按需解析的话可以拆分Json来读取,操作和解析方法复杂。

适用场景: 适用于需要处理超大型JSON文档、不需要对JSON文档进行按需解析、性能要求较高的场合。

解析过程:

  • 类似Gson,先创建1个对应于JSON数据的JavaBean类,再通过简单操作即可解析。
  • 与Gson解析不同的是:Gson可按需解析,即创建的JavaBean类不一定完全涵盖所要解析的JSON数据,按需创建属性;但Jackson解析对应的JavaBean必须把Json数据里面的所有key都有所对应,即必须把JSON内的数据所有解析出来,无法按需解析。

使用:

引入依赖

//Jackson
implementation 'com.fasterxml.jackson.core:jackson-databind:2.13.1'
implementation 'com.fasterxml.jackson.core:jackson-core:2.13.1'
implementation 'com.fasterxml.jackson.core:jackson-annotations:2.13.1'

/**
 * @Description: Jackson使用示例
 * @CreateDate: 2022/3/22 1:47 下午
 */
public class JacksonUse {
    public static void main(String[] args) throws Exception {
        Student student = new Student();
        student.setName("Lili");
        student.setSex("女");
        student.setAge(26);
        List<Course> courses = new ArrayList<>();
        courses.add(new Course("Art", 100));
        courses.add(new Course("Love", 99));
        student.setCourses(courses);

        ObjectMapper objectMapper = new ObjectMapper();
        //序列化 生成json文件
        File file = new File("json/jacksonjson.json");
        FileOutputStream fos = new FileOutputStream(file);
        objectMapper.writeValue(fos, student);

        //反序列化 json文件转为JavaBean
        Student student1 = objectMapper.readValue(file, Student.class);
        System.out.println(student1);
    }
}

注意:json里的key必须和JavaBean中的字段全部对应。

如果没对应,比如json中多了一个"hobby"字段,JavaBean中没有,运行就会报错:

Exception in thread "main" com.fasterxml.jackson.databind.exc.UnrecognizedPropertyException: Unrecognized field "hobby" (class site.exciter.learn.json.Student), not marked as ignorable (4 known properties: "courses", "name", "sex", "age"])
 at [Source: (File); line: 3, column: 13] (through reference chain: site.exciter.learn.json.Student["hobby"])

4 Fastjson解析

特点:

  • 快速FAST(比任何一款都快)
  • 面向对象
  • 功能强大(支持普通JDK类任意java bean Class,Collection,Map,Date或者enum)
  • 零依赖(只需要有JDK即可)
  • 支持注解,全类型序列化

使用:

引入依赖

implementation 'com.alibaba:fastjson:1.2.76'
/**
 * @Description: Fastjson使用示例
 * @CreateDate: 2022/3/22 2:10 下午
 */
public class FastjsonUse {
    public static void main(String[] args) throws Exception {
        Student student = new Student();
        student.setName("Lili");
        student.setSex("女");
        student.setAge(26);
        List<Course> courses = new ArrayList<>();
        courses.add(new Course("Art", 100));
        courses.add(new Course("Love", 99));
        student.setCourses(courses);

        //序列化 生成json
        File file = new File("json/fastjson.json");
        FileOutputStream fos = new FileOutputStream(file);
        JSONObject.writeJSONString(fos, student);

        //反序列化 json转JavaBean
        FileInputStream fis = new FileInputStream(file);
        Student student1 = JSONObject.parseObject(fis, Student.class);
        System.out.println(student1);
    }
}

以上就是几种JSON解析方式的对比,目前来说还是Gson和FastJson用的相对来说比较多。更多相关Android JSON内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Android 动态加载 so实现示例详解

    目录 背景 so动态加载介绍 从一个例子出发 so库检索与删除 动态加载so 结束了吗? ELF文件 扩展 总结 背景 对于一个普通的android应用来说,so库的占比通常都是巨高不下的,因为我们无可避免的在开发中遇到各种各样需要用到native的需求,所以so库的动态化可以减少极大的包体积,自从2020腾讯的bugly团队发部关于动态化so的相关文章后,已经过去两年了,相关文章,经过两年的考验,实际上so动态加载也是非常成熟的一项技术了. 但是很遗憾,许多公司都还没有这方面的涉略又或者说不知

  • Android 使用 Path 实现搜索动态加载动画效果

    今天实现一个搜索动态加载数据的动画效果,还是先看效果吧,用文字描述干巴巴的,看图说话什么都明白了, 实现这个就是使用Path中的getSegment()不断的去改变它截取片段的start和stop,再结合动画,今天就分步骤实现它,看完以后你也会觉的不是很难,只是没想到这么实现而已,所以要多见识,所谓眼界决定你的高度,还是延续我写博客的习惯,一步步分析,第一步就是绘制如下图: 如果单纯的绘制这个图很简单很简单的,绘制一个圆,然后再绘制一根线就搞定,但是要考虑这里的效果,就不能这么干了,如果你看了上

  • 浅谈Android Classloader动态加载分析

    ClassLoader概念 我们知道,Java源文件(.java)经过编译器编译之后,会转换成Java字节码(.class),然而程序是如何加载这些字节码文件到内存中呢?这就用到了ClassLoader,即类加载器.ClassLoader类加载器负责读取 Java 字节代码,并转换成 java.lang.Class类的一个实例.从而只有class文件被载入到了内存之后,才能被其程序所引用.所以ClassLoader就是用来动态加载class文件到内存当中用的. ClassLoader的分类 An

  • Android解析JSON格式数据的两种方式(JSONObject和Gson)

    目录 Json数据 JSONObject GSON 添加依赖 实际操作 实体类 总结 Json数据 接下来主要学习在Android中使用两种解析方式:JSONObject和Gson. JSONObject 比如我们有如下格式的Json数据: [{"id":"1","verison":"1.0","name":"shufu"}, {"id":"2",

  • Android NDK 开发中 SO 包大小压缩方法详解

    目录 背景 1.STL的使用方式 2.不使用Exception和RTTI RTTI Exception 3.使用 gc-sections去除没有用到的函数 4.去除冗余代码 5.设置编译器的优化flag 6.设置编译器的 Visibility Feature 7.设置编译器的Strip选项 8.去除C++代码中的iostream相关代码 总结 背景 这周在做Yoga包的压缩工作.Yoga本身是用BUCK脚本编译的,而最终编译出几个包大小大总共约为7M,不能满足项目中对于APK大小的限制,因此需要

  • Android列表组件ListView使用详解之动态加载或修改列表数据

    在使用ListView组件来显示列表数据时,有的时候我们需要改变列表中的数据,有以下方法: 1.重新给ListView组件设置适配器 这种方法重新创建了ListView,效率不好. 2.使用适配器中的方法 /** * Notifies the attached observers that the underlying data has been changed * and any View reflecting the data set should refresh itself. */ pu

  • Android中JSON的4种解析方式使用和对比

    目录 1 Android SDK自带的org.json解析 2 Gson解析 3 Jackson解析 4 Fastjson解析 1 Android SDK自带的org.json解析 解析原理: 基于文档驱动,需要把全部文件读入到内存中,然后遍历所有数据,根据需要检索想要的数据. 相关类: JSONObject JSONArray JSONTokener public Object nextValue() throws JSONException {     int c = nextCleanIn

  • Android 中倒计时验证两种常用方式实例详解

    Android 中倒计时验证两种常用方式实例详解 短信验证码功能,这里总结了两种常用的方式,可以直接拿来使用.看图: 说明:这里的及时从10开始,是为了演示的时间不要等太长而修改的. 1.第一种方式:Timer /** * Description:自定义Timer * <p> * Created by Mjj on 2016/12/4. */ public class TimeCount extends CountDownTimer { private Button button; //参数依

  • Android XML数据的三种解析方式

    本篇文章包含以下内容: XML数据的Dom解析      XML数据的Sax解析      XML数据的Pull解析      Activity中使用三种解析      Sax解析与Pull解析区别 三种解析方式的步骤: 1.在Assets文件夹中模拟创建XML数据 2.创建对应XML的Bean对象 3.开始解析 XML数据的Dom解析 DOM解析XML文件时,会将XML文件的所有内容读取到内存中(内存的消耗比较大),然后允许您使用DOM API遍历XML树.检索所需的数据 一.在Assets文

  • 深入Android中BroadcastReceiver的两种注册方式(静态和动态)详解

    今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式. BroadcastReceiver也就是"广播接收者"的意思,顾名思义,它就是用来接收来自系统和应用中的广播.在Android系统中,广播体现在方方面面,例如当开机完成后系统会产生一条广播,接收到这条广播就能实现开机启动服务的功能:当网络状态改变时系统会产生一条广播,接收到这条广播就能及时地做出提示和保存数据等操作:当电池电量改变时,系统会产生一条广播,接收到这条广播就能在电量低时告知用户

  • 详解Android中Fragment的两种创建方式

    fragment是Activity中用户界面的一个行为或者是一部分.你可以在一个单独的Activity上把多个Fragment组合成为一个多区域的UI,并且可以在多个Activity中再使用.你可以认为fragment是activity的一个模块零件,它有自己的生命周期,接收它自己的输入事件,并且可以在Activity运行时添加或者删除. 两个概念:Fragment.宿主 fragment的生命周期直接受其宿主activity的生命周期的影响.例如,一旦activity被暂停,它里面所有的fra

  • Android中Json数据读取与创建的方法

    首先介绍下JSON的定义,JSON是JavaScript Object Notation的缩写. 一种轻量级的数据交换格式,具有良好的可读和便于快速编写的特性.业内主流技术为其提供了完整的解决方案(有点类似于正则表达式,获得了当今大部分语言的支持),从而可以在不同平台间进行数据交换.JSON采用兼容性很高的文本格式,同时也具备类似于C语言体系的行为. JSON的结构: (1) Name/Value Pairs(无序的):类似所熟知的Keyed list. Hash table.Disctiona

  • javascript中JSON.parse()与eval()解析json的区别

    本文实例讲述了javascript中JSON.parse()与eval()解析json的区别.分享给大家供大家参考,具体如下: JSON(JavaScript Object Notation)是一种轻量级的数据格式,采用完全独立于语言的文本格式,是理想的数据交换格式.同时,JSON是Javascript原生格式,这意味着在javascript中处理JSON数据 基本格式: varjsonData='{"data1":"Hello,","data2"

  • Android中定时执行任务的3种实现方法(推荐)

    在Android开发中,定时执行任务的3种实现方法: 一.采用Handler与线程的sleep(long)方法(不建议使用,Java的实现方式) 二.采用Handler的postDelayed(Runnable, long)方法(最简单的android实现) 三.采用Handler与timer及TimerTask结合的方法(比较多的任务时建议使用) 下面逐一介绍: 一.采用Handle与线程的sleep(long)方法 Handler主要用来处理接受到的消息.这只是最主要的方法,当然Handle

  • Android BottomSheet效果的两种实现方式

    本文介绍了Android BottomSheet效果的两种实现方式,分享给大家,具体如下: BottomSheet效果 BottomSheet的效果是指从屏幕底部向上滑的效果,是MaterialDesign风格的一种,视觉效果如下: BottomSheet效果 实现这种效果有几种不同的方式,如果是在一个固定的页面上添加这种效果,可以在该页面布局中添加BoottomSheet相关的控件.如果是作为通用控件来提供给不同页面使用,则可以使用BottomSheetDialog实现,本文将对两种方法进行讲

  • Android中Activity的四种启动模式和onNewIntent()

    写在前面 Activity是Android四大组件之一,用于直接跟用户进行交互,本篇文章将介绍Activity的启动流程.用户启动Activity的方式大致有两种:一种是在桌面点击应用程序的图标,进入应用程序的主界面:另一种是在应用程序中,进入一个新的Activity.前者,桌面其实是系统应用launcher的界面,点击应用程序图标,会进行应用程序的主界面,实质是从一个应用的Activity进入另一个应用Activity. 因此,不管是从桌面进入应用主界面,还是在应用里进入一个新的Activit

随机推荐