編輯:關於Android編程
本文實例分析了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
android2.3提供的json解析類
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文本
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:
構造函數
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':01,'name':'tom','gender':'男'}}
多個數據
{"singers":[ {'id':02,'name':'tom','gender':'男'}, {'id':03,'name':'jerry,'gender':'男'}, {'id':04,'name':'jim,'gender':'男'}, {'id':05,'name':'lily,'gender':'女'}]}
下面的類主要是解析單個數據parseJson()和多個數據的方法parseJsonMulti():
public class JsonActivity extends Activity { /** Called when the activity is first created. */ 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 = jsonObjs.getJSONObject(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(); } } }
希望本文所述對大家Android程序設計有所幫助。
背景上周發現蘑菇街IM-Android代碼裡面,一些地方代碼編寫不當,存在內存洩漏的問題,在和瘋紫交流的過程中,發現加深了一些理解,所以決定寫一下分析思路,相互學習。內存
一.ViewPager頁面滑動切換實現:借鑒了別人的源碼,還是比較容易實現的,而且這種效果還是經常使用的,特此記錄一下:效果圖:主界面: public clas
Android插件開發初探對於Android的插件化其實已經討論已久了,但是市面上還沒有非常靠譜成熟的插件框架供我們使用。這裡我們就嘗試性的對比一下Java中,我們使用插
平時的一些Android學習視頻中,他們都是基於Android2.2的去使用ListView,我看到都是會在UI線程中去訪問網絡獲取數據,但是這在Android4.0之後