首页 > android Json解析详解

android Json解析详解

JSON的定义:

       一种轻量级的数据交换格式,具有良好的可读和便于快速编写的特性。业内主流技术为其提供了完整的解决方案(有点类似于正则表达式 ,获得了当今大部分语 言的支持),从而可以在不同平台间进行数据交换。JSON采用兼容性很高的文本格式,同时也具备类似于C语言体系的行为– Json.org

JSON Vs XML

1.JSON和XML的数据可读性基本相同

2.JSON和XML同样拥有丰富的解析手段

3.JSON相对于XML来讲,数据的体积小

4.JSON与JavaScript的交互更加方便

5.JSON对数据的描述性比XML较差

6.JSON的速度要远远快于XML



pull解析xml详解:http://blog.csdn.net/onlyonecoder/article/details/8489170



android2.3提供的json解析类 

android的json解析部分都在包org.json下,主要有以下几个类: 

JSONObject:可以看作是一个json对象,这是系统中有关JSON定义的基本单元,其包含一对儿(Key/Value)数值。它对外部(External:   应用toString()方法输出的数值)调用的响应体现为一个标准的字符串(例如:{"JSON": "Hello, World"},最外被大括号包裹,其中的KeyValue被冒号":"分隔)。其对于内部(Internal)行为的操作格式略微,例如:初始化一个JSONObject实例,引用内部的put()方法添加数值:new JSONObject().put("JSON", "Hello, World!"),在KeyValue之间是以逗号","分隔。Value的类型包括:BooleanJSONArrayJSONObjectNumberString或者默认值JSONObject.NULL object 。

JSONStringer:json 文本构建类 ,根据官方的解释,这个类可以帮助快速和便捷的创建JSON text。其最大的优点在于可以减少由于 格式的错误导致程序异常,引用这个类可以自动严格按照JSON语法规则(syntax rules)创建JSON text。每个JSONStringer实体只能对应创建一个JSON text。。其最大的优点在于可以减少由于格式的错误导致程序异常,引用这个类可以自动严格按照JSON语法规则(syntax rules)创建JSON text。每个JSONStringer实体只能对应创建一个JSON text。

JSONArray: 它代表一组有序的数值。将其转换为String输出(toString)所表现的形式是用方括号包裹,数值以逗号”,”分隔(例如:     [value1,value2,value3],大家可以亲自利用简短的代码更加直观的了解其格式)。这个类的内部同样具有查询行为,     get()和opt()两种方法都可以通过index索引返回指定的数值,put()方法用来添加或者替换数值。同样这个类的value类型可以包 括:Boolean、JSONArray、JSONObject、Number、String或者默认值JSONObject.NULL object。

JSONTokener:json解析类 

JSONException:json中用到的异常 





JSONObject, JSONArray来构建json文本  

 

[java] view plaincopy
  1. // 假设现在要创建这样一个json文本    
  2. //  {    
  3. //      "phone" : ["12345678", "87654321"], // 数组    
  4. //      "name" : "yuanzhifei89", // 字符串    
  5. //      "age" : 100, // 数值    
  6. //      "address" : { "country" : "china", "province" : "jiangsu" }, // 对象    
  7. //      "married" : false // 布尔值    
  8. //  }    
  9.     
  10. try {    
  11.     // 首先最外层是{},是创建一个对象    
  12.     JSONObject person = new JSONObject();    
  13.     // 第一个键phone的值是数组,所以需要创建数组对象    
  14.     JSONArray phone = new JSONArray();    
  15.     phone.put("12345678").put("87654321");    
  16.     person.put("phone", phone);    
  17.     
  18.     person.put("name", "yuanzhifei89");    
  19.     person.put("age", 100);    
  20.     // 键address的值是对象,所以又要创建一个对象    
  21.     JSONObject address = new JSONObject();    
  22.     address.put("country", "china");    
  23.     address.put("province", "jiangsu");    
  24.     person.put("address", address);      
  25.     person.put("married", false);    
  26. catch (JSONException ex) {    
  27.     // 键为null或使用json不支持的数字格式(NaN, infinities)    
  28.     throw new RuntimeException(ex);    
  29. }    





getType和optType api的使用   

 

getType可以将要获取的键的值转换为指定的类型,如果无法转换或没有值则抛出JSONException 

optType也是将要获取的键的值转换为指定的类型,无法转换或没有值时返回用户提供或这默认提供的值 

 

[java] view plaincopy
  1. try {    
  2.     // 所有使用的对象都是用上面创建的对象    
  3.     // 将第一个电话号码转换为数值和将名字转换为数值    
  4.     phone.getLong(0);    
  5.     person.getLong("name"); // 会抛异常,因为名字无法转换为long          
  6.     phone.optLong(0); // 代码内置的默认值    
  7.     phone.optLong(0, 1000); // 用户提供的默认值    
  8.     person.optLong("name");    
  9.     person.optLong("name", 1000); // 不像上面那样抛异常,而是返回1000    
  10. catch (JSONException ex) {    
  11.     // 异常处理代码    
  12. }    




 



除了上面的两个类,还可以使用JSONStringer来构建json文本  

 

Java代码  



[java] view plaincopy
  1. try {    
  2.     JSONStringer jsonText = new JSONStringer();    
  3.     // 首先是{,对象开始。object和endObject必须配对使用    
  4.     jsonText.object();    
  5.         
  6.     jsonText.key("phone");    
  7.     // 键phone的值是数组。array和endArray必须配对使用    
  8.     jsonText.array();    
  9.     jsonText.value("12345678").value("87654321");    
  10.     jsonText.endArray();    
  11.         
  12.     jsonText.key("name");    
  13.     jsonText.value("yuanzhifei89");    
  14.     jsonText.key("age");    
  15.     jsonText.value(100);    
  16.         
  17.     jsonText.key("address");    
  18.     // 键address的值是对象    
  19.     jsonText.object();    
  20.     jsonText.key("country");    
  21.     jsonText.value("china");    
  22.     jsonText.key("province");    
  23.     jsonText.value("jiangsu");    
  24.     jsonText.endObject();    
  25.         
  26.     jsonText.key("married");    
  27.     jsonText.value(false);    
  28.         
  29.     // },对象结束    
  30.     jsonText.endObject();    
  31. catch (JSONException ex) {    
  32.     throw new RuntimeException(ex);    
  33. }    





json文本解析类JSONTokener   

按照RFC4627规范将json文本解析为相应的对象。  



对于将json文本解析为对象,只需要用到该类的两个api:   

构造函数  

public Object nextValue(); 

 

[java] view plaincopy
  1. //  {    
  2. //      "phone" : ["12345678", "87654321"], // 数组    
  3. //      "name" : "yuanzhifei89", // 字符串    
  4. //      "age" : 100, // 数值    
  5. //      "address" : { "country" : "china", "province" : "jiangsu" }, // 对象    
  6. //      "married" : false // 布尔值    
  7. //  }    
  8.     
  9. private static final String JSON =     
  10. "{" +    
  11.     "   "phone" : ["12345678", "87654321"]," +    
  12.     "   "name" : "yuanzhifei89"," +    
  13.     "   "age" : 100," +    
  14.     "   "address" : { "country" : "china", "province" : "jiangsu" }," +    
  15.     "   "married" : false," +    
  16. "}";    
  17.     
  18. try {    
  19.     JSONTokener jsonParser = new JSONTokener(JSON);    
  20.     // 此时还未读取任何json文本,直接读取就是一个JSONObject对象。    
  21.     // 如果此时的读取位置在"name" : 了,那么nextValue就是"yuanzhifei89"(String)    
  22.     JSONObject person = (JSONObject) jsonParser.nextValue();    
  23.     // 接下来的就是JSON对象的操作了    
  24.     person.getJSONArray("phone");    
  25.     person.getString("name");    
  26.     person.getInt("age");    
  27.     person.getJSONObject("address");    
  28.     person.getBoolean("married");    
  29. catch (JSONException ex) {    
  30.     // 异常处理代码    
  31. }    





 

其它的api基本就是用来查看json文本中的文本的 

 

[java] view plaincopy
  1. try {    
  2.     JSONTokener jsonParser = new JSONTokener(JSON);    
  3.     // 继续向下读8个json文本中的字符。此时刚开始,即在{处    
  4.     jsonParser.next(8); //{    "phone。tab算一个字符    
  5.         
  6.     // 继续向下读1个json文本中的字符    
  7.     jsonParser.next(); //"    
  8.         
  9.     // 继续向下读取一个json文本中的字符。该字符不是空白、同时也不是注视中的字符    
  10.     jsonParser.nextClean(); //:    
  11.         
  12.     // 返回当前的读取位置到第一次遇到'a'之间的字符串(不包括a)。    
  13.     jsonParser.nextString('a'); //  ["12345678", "87654321"],    "n(前面有两个空格)    
  14.         
  15.     // 返回当前读取位置到第一次遇到字符串中(如"0089")任意字符之间的字符串,同时该字符是trimmed的。(此处就是第一次遇到了89)    
  16.     jsonParser.nextTo("0089"); //me" : "yuanzhifei    
  17.         
  18.     // 读取位置撤销一个    
  19.     jsonParser.back();    
  20.     jsonParser.next(); //i    
  21.         
  22.     // 读取位置前进到指定字符串处(包括字符串)    
  23.     jsonParser.skipPast("address");    
  24.     jsonParser.next(8); //" : { "c    
  25.         
  26.     // 读取位置前进到执行字符处(不包括字符)    
  27.     jsonParser.skipTo('m');    
  28.     jsonParser.next(8); //married"    
  29. catch (JSONException ex) {    
  30.     // 异常处理代码    
  31. }    





以下是一个标准的JSON请求实现过程:

 

[java] view plaincopy
  1. HttpPost request = new HttpPost(url);   
  2. // 先封装一个 JSON 对象   
  3. JSONObject param = new JSONObject();   
  4. param.put("name", "rarnu");   
  5. param.put("password", "123456");   
  6. // 绑定到请求 Entry   
  7. StringEntity se = new StringEntity(param.toString());    
  8. request.setEntity(se);   
  9. // 发送请求   
  10. HttpResponse httpResponse = new DefaultHttpClient().execute(request);   
  11. // 得到应答的字符串,这也是一个 JSON 格式保存的数据   
  12. String retSrc = EntityUtils.toString(httpResponse.getEntity());   
  13. // 生成 JSON 对象   
  14. JSONObject result = new JSONObject( retSrc);   
  15. String token = result.get("token");  





下面这个是自己修改别人的小例子,主要是加一些注释和讲解,这个例子主要是使用android进行json解析。

  • 多个数据{ "singers":[   
  •         { 'id':02,'name':'tom','gender':'男'},   
  •          { 'id':03,'name':'jerry,'gender':'男'},   
  • { 'id':04,'name':'jim,'gender':'男'},   
  • { 'id':05,'name':'lily,'gender':'女'}]}                                                                              



                                             

  下面的类主要是解析单个数据parseJson()和多个数据的方法parseJsonMulti():

 

[java] view plaincopy
    1. public class JsonActivity extends Activity {   
    2.     /** Called when the activity is first created. */   
    3.     private TextView tvJson;   
    4.     private Button btnJson;   
    5.     private Button btnJsonMulti;   
    6.     @Override   
    7.     public void onCreate(Bundle savedInstanceState) {   
    8.         super.onCreate(savedInstanceState);   
    9.         setContentView(R.layout.main);   
    10.         tvJson = (TextView) this.findViewById(R.id.tvJson);   
    11.         btnJson = (Button) this.findViewById(R.id.btnJson);   
    12.         btnJsonMulti = (Button) this.findViewById(R.id.btnJsonMulti);   
    13.         btnJson.setOnClickListener(new View.OnClickListener() {   
    14.             @Override   
    15.             public void onClick(View v) {   
    16.                 // url   
    17.                 // String strUrl = "http://10.158.166.110:8080/AndroidServer/JsonServlet";   
    18.                 String strUrl = ServerPageUtil.getStrUrl(UrlsOfServer.JSON_SINGER);   
    19.                 //获得返回的Json字符串   
    20.                 String strResult = connServerForResult(strUrl);   
    21.                 //解析Json字符串   
    22.                 parseJson(strResult);   
    23.             }   
    24.         });   
    25.         btnJsonMulti.setOnClickListener(new View.OnClickListener() {   
    26.             @Override   
    27.             public void onClick(View v) {   
    28.                 String strUrl = ServerPageUtil.getStrUrl(UrlsOfServer.JSON_SINGERS);   
    29.                 String strResult = connServerForResult(strUrl);   
    30.                 //获得多个Singer   
    31.                 parseJsonMulti(strResult);   
    32.             }   
    33.         });   
    34.     }   
    35.     private String connServerForResult(String strUrl) {   
    36.         // HttpGet对象   
    37.         HttpGet httpRequest = new HttpGet(strUrl);   
    38.         String strResult = "";   
    39.         try {   
    40.             // HttpClient对象   
    41.             HttpClient httpClient = new DefaultHttpClient();   
    42.             // 获得HttpResponse对象   
    43.             HttpResponse httpResponse = httpClient.execute(httpRequest);   
    44.             if (httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {   
    45.                 // 取得返回的数据   
    46.                 strResult = EntityUtils.toString(httpResponse.getEntity());   
    47.             }   
    48.         } catch (ClientProtocolException e) {   
    49.             tvJson.setText("protocol error");   
    50.             e.printStackTrace();   
    51.         } catch (IOException e) {   
    52.             tvJson.setText("IO error");   
    53.             e.printStackTrace();   
    54.         }   
    55.         return strResult;   
    56.     }   
    57.     // 普通Json数据解析   
    58.     private void parseJson(String strResult) {   
    59.         try {   
    60.             JSONObject jsonObj = new JSONObject(strResult).getJSONObject("singer");   
    61.             int id = jsonObj.getInt("id");   
    62.             String name = jsonObj.getString("name");   
    63.             String gender = jsonObj.getString("gender");   
    64.             tvJson.setText("ID号"+id + ", 姓名:" + name + ",性别:" + gender);   
    65.         } catch (JSONException e) {   
    66.             System.out.println("Json parse error");   
    67.             e.printStackTrace();   
    68.         }   
    69.     }   
    70.     //解析多个数据的Json  
    71.     private void parseJsonMulti(String strResult) {   
    72.         try {   
    73.             JSONArray jsonObjs = new JSONObject(strResult).getJSONArray("singers");   
    74.             String s = "";   
    75.             for(int i = 0; i < jsonObjs.length() ; i++){   
    76.                 JSONObject jsonObj = ((JSONObject)jsonObjs.opt(i))   
    77.                 .getJSONObject("singer");   
    78.                 int id = jsonObj.getInt("id");   
    79.                 String name = jsonObj.getString("name");   
    80.                 String gender = jsonObj.getString("gender");   
    81.                 s +=  "ID号"+id + ", 姓名:" + name + ",性别:" + gender+ " " ;   
    82.             }   
    83.             tvJson.setText(s);   
    84.         } catch (JSONException e) {   
    85.             System.out.println("Jsons parse error !");   
    86.             e.printStackTrace();   
    87.         }   
    88.     }   
    89. }  

转载于:https://www.cnblogs.com/Free-Thinker/p/3862271.html

更多相关:

  • importjava.security.SecureRandom;importjavax.crypto.Cipher;importjavax.crypto.SecretKey;importjavax.crypto.SecretKeyFactory;importjavax.crypto.spec.DESKeySpec;//结果与DES算...

  • 题目:替换空格 请实现一个函数,把字符串 s 中的每个空格替换成"%20"。 输入:s = "We are happy." 输出:"We%20are%20happy." 限制: 0 <= s 的长度 <= 10000 解题: 时间复杂度:O(n) 空间复杂度:O(n) class Solution { public:s...

  • 在C++11标准库中,string.h已经添加了to_string方法,方便从其他类型(如整形)快速转换成字面值。 例如: for (size_t i = 0; i < texArrSize; i++)RTX_Shader.SetInt(string("TexArr[") + to_string(i) + "]", 7 + i);...

  • Ubuntu 14.04安装并升级之后,变成楷体字体非常难看,我昨天搞了一晚上,终于理了个头绪,这里整理一下。 经过网上调研,大家的一致看法是,使用开源字体库文泉驿的微黑字体效果比较理想,甚至效果不输windows平台的雅黑字体。下面我打算微黑来美化Ubuntu 14.04. 1.安装文泉驿微黑字体库 sudo aptitude...

  • 使用string时发现了一些坑。 我们知道stl 容器并不是线程安全的,所以在使用它们的过程中往往需要一些同步机制来保证并发场景下的同步更新。 应该踩的坑还是一个不拉的踩了进去,所以还是记录一下吧。 string作为一个容器,随着我们的append 或者 针对string的+ 操作都会让string内部的数据域动态增加,而动态增加的...

  • 点云PCL免费知识星球,点云论文速读。文章:DSP-SLAM: Object Oriented SLAM with Deep Shape Priors作者:Jingwen Wang Martin Runz Lourdes Agapito编译:点云PCL代码:https://github.com/JingwenWang95/DSP-S...

  • RAM缓存 新RAM缓存算法(CLFUS) 新的RAM缓存使用的创意来自许多缓存替换策略和算法,包括LRU,LFU,CLOCK,GDFS及2Q,它被命名为时钟周期内最小频繁使用大小算法CLFUS(Clocked Least Frequently Used by Size)。它避开了任何专利算法,具有如下特性: 均衡最近性(Rec...

  • MP4 |视频:AVC,1280×720 30 fps |音频:AAC,48 KHz,2 Ch |时长:2h 12m 语言:英语+中英文字幕(根据原英文字幕机译更准确)|大小解压后:560M C4D是一个有抱负的运动图形艺术家和设计师的重要工具。借助C4D,您可以使用3D对象、动态效果和动画来增强运动图形、模型和可视化效果。本课...

  • 文章目录先说问题:再说解决尝试1:尝试2(该尝试建议先在自己环境搭配对应业务测试通过后再现场尝试): 感谢 学无止境996同学的陪伴和vigourtyy美丽女友的支持,直到这个解决问题的深夜 先说问题: ceph 12.2.1生产环境:3副本 tier + 3副本data 机房在拥有业务的情况下重启集群交换机,产生如下场景...

  • 这周主要学习了java中的类和对象的知识点,发现和C++中的类和对象极为相似,对于类和对象的概念理解起来也简单。同时在自学的过程中也把类的知识重新复习巩固了一下(如类的三大特征:继承,封装和多态,构造,成员对象的访问权限,构造,无参有参函数的调用等),同时也了解到一些新的概念,比如类对象创建和引用占据堆内存和栈内存,输出对象时默认调...