Json解析

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Json解析相关的知识,希望对你有一定的参考价值。

什么是json

json(javascript Object Notation)是一种轻量级的数据交换格式,相比于xml这种数据交换格式来说,因为解析xml比较的复杂,而且需要编写大段的代码,所以客户端和服务器的数据交换格式往往通过json来进行交换。尤其是对于web开发来说,json数据格式在客户端直接可以通过javascript来进行解析。


json一共有两种数据结构,一种是以 (key/value)对形式存在的无序的jsonObject对象,一个对象以“{”(左花括号)开始,“}”(右花括号)结束。每个“名称”后跟一个“:”(冒号);“‘名称/值’ 对”之间使用“,”(逗号)分隔。


例如:{"name": "xiaoluo"}, 这就是一个最简单的json对象,对于这种数据格式,key值必须要是string类型,而对于value,则可以是string、number、object、array、true、false、null等数据类型:


另一种数据格式就是有序的value的集合,这种形式被称为是jsonArray,数组是值(value)的有序集合。一个数组以“[”(左中括号)开始,“]”(右中括号)结束。值之间使用“,”(逗号)分隔。  

    

    

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


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

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


// 假设现在要创建这样一个json文本    

//  {    

//      "phone" : ["12345678", "87654321"], // 数组    

//      "name" : "yuanzhifei89", // 字符串    

//      "age" : 100, // 数值    

//      "address" : { "country" : "china", "province" : "jiangsu" }, // 对象    

//      "married" : false // 布尔值    

//  }    

    

try {    

    // 首先最外层是{},是创建一个对象    

    JSONObject person = new JSONObject();    

    // 第一个键phone的值是数组,所以需要创建数组对象    

    JSONArray phone = new JSONArray();

    phone.put("12345678").put("87654321");

    person.put("phone", phone);

    

    person.put("name", "yuanzhifei89");

    person.put("age", 100);

    // 键address的值是对象,所以又要创建一个对象

    JSONObject address = new JSONObject();

    address.put("country", "china");

    address.put("province", "jiangsu");

    person.put("address", address);      

    person.put("married", false);    

} catch (JSONException ex) {    

    // 键为null或使用json不支持的数字格式(NaN, infinities)    

    throw new RuntimeException(ex);    

}    



getType和optType api的使用   

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

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


try {

    // 所有使用的对象都是用上面创建的对象

    // 将第一个电话号码转换为数值和将名字转换为数值

    phone.getLong(0);

    person.getLong("name"); // 会抛异常,因为名字无法转换为long

    phone.optLong(0); // 代码内置的默认值

    phone.optLong(0, 1000); // 用户提供的默认值

    person.optLong("name");

    person.optLong("name", 1000); // 不像上面那样抛异常,而是返回1000

} catch (JSONException ex) {

    // 异常处理代码

}    




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

Java代码


try {

    JSONStringer jsonText = new JSONStringer();

    // 首先是{,对象开始。object和endObject必须配对使用

    jsonText.object();

        

    jsonText.key("phone");

    // 键phone的值是数组。array和endArray必须配对使用

    jsonText.array();

    jsonText.value("12345678").value("87654321");

    jsonText.endArray();

        

    jsonText.key("name");

    jsonText.value("yuanzhifei89");

    jsonText.key("age");

    jsonText.value(100);

        

    jsonText.key("address");

    // 键address的值是对象

    jsonText.object();

    jsonText.key("country");

    jsonText.value("china");

    jsonText.key("province");

    jsonText.value("jiangsu");

    jsonText.endObject();

    

    jsonText.key("married");

    jsonText.value(false);

        

    // },对象结束    

    jsonText.endObject();

} catch (JSONException ex) {

    throw new RuntimeException(ex);

}    



json文本解析类JSONTokener   

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


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

构造函数JSONTokener(String in)

public Object nextValue(); 


//  {    

//      "phone" : ["12345678", "87654321"], // 数组    

//      "name" : "yuanzhifei89", // 字符串    

//      "age" : 100, // 数值    

//      "address" : { "country" : "china", "province" : "jiangsu" }, // 对象    

//      "married" : false // 布尔值    

//  }    

    

private static final String JSON =     

"{" +    

    "   \"phone\" : [\"12345678\", \"87654321\"]," +    

    "   \"name\" : \"yuanzhifei89\"," +    

    "   \"age\" : 100," +    

    "   \"address\" : { \"country\" : \"china\", \"province\" : \"jiangsu\" }," +    

    "   \"married\" : false," +    

"}";    

    

try {    

    JSONTokener jsonParser = new JSONTokener(JSON);    

    // 此时还未读取任何json文本,直接读取就是一个JSONObject对象。    

    // 如果此时的读取位置在"name" : 了,那么nextValue就是"yuanzhifei89"(String)    

    JSONObject person = (JSONObject) jsonParser.nextValue();    

    // 接下来的就是JSON对象的操作了    

    person.getJSONArray("phone");    

    person.getString("name");    

    person.getInt("age");    

    person.getJSONObject("address");    

    person.getBoolean("married");    

} catch (JSONException ex) {    

    // 异常处理代码    

}    



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


try {    

    JSONTokener jsonParser = new JSONTokener(JSON);    

    // 继续向下读8个json文本中的字符。此时刚开始,即在{处    

    jsonParser.next(8); //{    "phone。tab算一个字符    

        

    // 继续向下读1个json文本中的字符    

    jsonParser.next(); //"    

        

    // 继续向下读取一个json文本中的字符。该字符不是空白、同时也不是注视中的字符    

    jsonParser.nextClean(); //:    

        

    // 返回当前的读取位置到第一次遇到‘a‘之间的字符串(不包括a)。    

    jsonParser.nextString(‘a‘); //  ["12345678", "87654321"],    "n(前面有两个空格)    

        

    // 返回当前读取位置到第一次遇到字符串中(如"0089")任意字符之间的字符串,同时该字符是trimmed的。(此处就是第一次遇到了89)    

    jsonParser.nextTo("0089"); //me" : "yuanzhifei    

        

    // 读取位置撤销一个    

    jsonParser.back();    

    jsonParser.next(); //i    

        

    // 读取位置前进到指定字符串处(包括字符串)    

    jsonParser.skipPast("address");    

    jsonParser.next(8); //" : { "c    

        

    // 读取位置前进到执行字符处(不包括字符)    

    jsonParser.skipTo(‘m‘);    

    jsonParser.next(8); //married"    

} catch (JSONException ex) {    

    // 异常处理代码    

}    



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


HttpPost request = new HttpPost(url);   

// 先封装一个 JSON 对象   

JSONObject param = new JSONObject();   

param.put("name", "rarnu");   

param.put("password", "123456");   

// 绑定到请求 Entry   

StringEntity se = new StringEntity(param.toString());    

request.setEntity(se);   

// 发送请求   

HttpResponse httpResponse = new DefaultHttpClient().execute(request);   

// 得到应答的字符串,这也是一个 JSON 格式保存的数据   

String retSrc = EntityUtils.toString(httpResponse.getEntity());   

// 生成 JSON 对象   

JSONObject result = new JSONObject( retSrc);   

String token = result.get("token");  



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


单数据{‘singer‘:{‘id‘:1,‘name‘:‘tom‘,‘gender‘:‘男‘}}   

多个数据{‘singers‘:[{‘id‘:‘2‘,‘name‘:‘tom‘,‘gender‘:‘男‘},{‘id‘:‘3‘,‘name‘:‘jerry‘,‘gender‘:‘男‘},{‘id‘:‘4‘,‘name‘:‘jim‘,‘gender‘:‘男‘},{‘id‘:‘5‘,‘name‘:‘lily‘,‘gender‘:‘女‘}]}                                                                              


                                             

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


public class JsonActivity extends Activity {

    private TextView tvJson;   

    private Button btnJson;   

    private Button btnJsonMulti;   

    @Override   

    public void onCreate(Bundle savedInstanceState) {   

        super.onCreate(savedInstanceState);   

        setContentView(R.layout.main);   

        tvJson = (TextView) this.findViewById(R.id.tvJson);   

        btnJson = (Button) this.findViewById(R.id.btnJson);   

        btnJsonMulti = (Button) this.findViewById(R.id.btnJsonMulti);   

        btnJson.setOnClickListener(new View.OnClickListener() {   

            @Override   

            public void onClick(View v) {   

                // url   

                // String strUrl = "http://10.158.166.110:8080/AndroidServer/JsonServlet";

                String strUrl = ServerPageUtil.getStrUrl(UrlsOfServer.JSON_SINGER);

                //获得返回的Json字符串

                String strResult = connServerForResult(strUrl);

                //解析Json字符串

                parseJson(strResult);

            }

        });

        btnJsonMulti.setOnClickListener(new View.OnClickListener() {

            @Override

            public void onClick(View v) {

                String strUrl = ServerPageUtil.getStrUrl(UrlsOfServer.JSON_SINGERS);

                String strResult = connServerForResult(strUrl);

                //获得多个Singer

                parseJsonMulti(strResult);

            }

        });

    }   

    private String connServerForResult(String strUrl) {   

        // HttpGet对象   

        HttpGet httpRequest = new HttpGet(strUrl);   

        String strResult = "";   

        try {   

            // HttpClient对象   

            HttpClient httpClient = new DefaultHttpClient();   

            // 获得HttpResponse对象   

            HttpResponse httpResponse = httpClient.execute(httpRequest);   

            if (httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {   

                // 取得返回的数据   

                strResult = EntityUtils.toString(httpResponse.getEntity());   

            }   

        } catch (ClientProtocolException e) {   

            tvJson.setText("protocol error");   

            e.printStackTrace();   

        } catch (IOException e) {   

            tvJson.setText("IO error");   

            e.printStackTrace();   

        }   

        return strResult;   

    }   

    // 普通Json数据解析   

    private void parseJson(String strResult) {   

        try {   

            JSONObject jsonObj = new JSONObject(strResult).getJSONObject("singer");   

            int id = jsonObj.getInt("id");   

            String name = jsonObj.getString("name");   

            String gender = jsonObj.getString("gender");   

            tvJson.setText("ID号"+id + ", 姓名:" + name + ",性别:" + gender);   

        } catch (JSONException e) {   

            System.out.println("Json parse error");   

            e.printStackTrace();   

        }   

    }   

    //解析多个数据的Json  

    private void parseJsonMulti(String strResult) {   

        try {   

            JSONArray jsonObjs = new JSONObject(strResult).getJSONArray("singers");   

            String s = "";   

            for(int i = 0; i < jsonObjs.length() ; i++){   

                JSONObject jsonObj = (JSONObject)jsonObjs.get(i);   

                int id = jsonObj.getInt("id");   

                String name = jsonObj.getString("name");   

                String gender = jsonObj.getString("gender");   

                s +=  "ID号"+id + ", 姓名:" + name + ",性别:" + gender+ "\n" ;   

            }   

            tvJson.setText(s);   

        } catch (JSONException e) {   

            System.out.println("Jsons parse error !");   

            e.printStackTrace();   

        }   

    }   

}   


以上是关于Json解析的主要内容,如果未能解决你的问题,请参考以下文章

使用 json rereiver php mysql 在片段中填充列表视图

ios - Heroku 和 MongoDb 上的自定义解析服务器错误 3080:JSON 文本没有以数组或对象开头,并且允许未设置片段的选项

NodeJs GraphQL 片段解析器

不要在片段中显示列表视图项

Json解析工具类

Golang实践录:反射reflect的一些研究及代码汇总