Android系列---JSON資料解析

xiaoluo501395377發表於2013-11-27
您可以通過點選 右下角 的按鈕 來對文章內容作出評價, 也可以通過左下方的 關注按鈕 來關注我的部落格的最新動態。 

如果文章內容對您有幫助, 不要忘記點選右下角的 推薦按鈕 來支援一下哦   

如果您對文章內容有任何疑問, 可以通過評論或發郵件的方式聯絡我: 501395377@qq.com  / lzp501395377@gmail.com

如果需要轉載,請註明出處,謝謝!!

上篇隨筆詳細介紹了三種解析伺服器端傳過來的xml資料格式,而對於伺服器端來說,返回給客戶端的資料格式一般分為html、xml和json這三種格式,那麼本篇隨筆將講解一下json這個知識點,包括如何通過json-lib和gson這兩個json解析庫來對解析我們的json資料,以及如何在我們的Android客戶端解析來自伺服器端的json資料,並更新到UI當中。

一、什麼是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等資料型別:

另一種資料格式就是有序的value的集合,這種形式被稱為是jsonArray,陣列是值(value)的有序集合。一個陣列以“[”(左中括號)開始,“]”(右中括號)結束。值之間使用“,”(逗號)分隔。

 

更多的有關json資料格式可以參加json的官網,http://www.json.org/json-zh.html

二、解析json資料格式

這裡將使用兩種json的解析庫來對我們的json資料格式進行解析以及生成我們的json資料格式。

1.json-lib(http://json-lib.sourceforge.net/)

使用json-lib來進行解析,我們需要引入第三方的包,因為json-lib分為了兩個版本,一個版本是針對於jdk1.3的,一個版本是針對於jdk1.5的,這裡我們下載jdk1.5的這個json-lib包,其中還需要引入其他的幾個jar包:

下載好這幾個jar包後,加入到classpath中即可。我們來看看json-lib給我們提供的API。

我們最常用的兩個類就是  JSONObject和JSONArray這兩個類,分別代表了json物件和json陣列,這兩個類都實現了 JSON 這個介面,下面我們通過幾個小例子來看看如何將我們常見的幾種資料格式轉換成我們的json物件(我們一般稱之為JSON資料的序列化)以及再將json物件在轉換成我們的資料格式(稱之為反序列化)。

①簡單的javabean的序列化和反序列化

public class Person
{
    private int id;
    private String name;
    private String address;

    public Person()
    {
    }

    public int getId()
    {
        return id;
    }

    public void setId(int id)
    {
        this.id = id;
    }

    public String getName()
    {
        return name;
    }

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

    public String getAddress()
    {
        return address;
    }

    public void setAddress(String address)
    {
        this.address = address;
    }

    public Person(int id, String name, String address)
    {
        super();
        this.id = id;
        this.name = name;
        this.address = address;
    }

    @Override
    public String toString()
    {
        return "Person [id=" + id + ", name=" + name + ", address=" + address
                + "]";
    }

}

首先我們定義一個簡單的javabean物件,然後將一個Person物件轉換成json物件,然後再將這個json物件反序列化成我們的Person物件。

我們先定義一個JsonTools類,這個類有兩個靜態方法,我們可以通過這兩個方法來得到一個JSON型別的字串物件,以及一個JSON物件

public class JsonTools
{
    /**
     * 得到一個json型別的字串物件
     * @param key
     * @param value
     * @return
     */
    public static String getJsonString(String key, Object value)
    {
        JSONObject jsonObject = new JSONObject();
        //put和element都是往JSONObject物件中放入 key/value 對
//        jsonObject.put(key, value);
        jsonObject.element(key, value);
        return jsonObject.toString();
    }
    
    /**
     * 得到一個json物件
     * @param key
     * @param value
     * @return
     */
    public static JSONObject getJsonObject(String key, Object value)
    {
        JSONObject jsonObject = new JSONObject();
        jsonObject.put(key, value);
        return jsonObject;
    }
    
}

我們可以直接通過 JSONObject jsonObject = new JSONObject(); 這個方法就可以得到一個json物件,然後通過element()或者是put()方法來給我們的json物件新增key/value對。我們先來看看第一個例子,實現一個簡單的Person物件和json物件的轉換

     Person person = new Person(1, "xiaoluo", "廣州");
        //    將Person物件轉換成一個json型別的字串物件
        String personString = JsonTools.getJsonString("person", person);
        System.out.println(personString.toString());

我們看看控制檯的輸出:

{"person":{"address":"廣州","id":1,"name":"xiaoluo"}}

整個外面的大括號是一個json物件,裡面有一對key/value,其中裡面的{"address":"廣州","id":1,"name":"xiaoluo"}就是我們轉換成的json字串物件

再來看看如何將json物件轉換成我們的bean物件

     JSONObject jsonObject = JsonTools.getJsonObject("person", person);
        //    通過JSONObject的toBean方法可以將json物件轉換成一個javabean
        JSONObject personObject = jsonObject.getJSONObject("person");
        Person person2 = (Person) JSONObject.toBean(personObject, Person.class);
        System.out.println(person2);
Person [id=1, name=xiaoluo, address=廣州]

②轉換List<Person>型別的物件

  @Test
    public void testPersonsJson()
    {
        List<Person> persons = new ArrayList<Person>();
        Person person = new Person(1, "xiaoluo", "廣州");
        Person person2 = new Person(2, "android", "上海");
        persons.add(person);
        persons.add(person2);
        String personsString = JsonTools.getJsonString("persons", persons);
        System.out.println(personsString);
        
        JSONObject jsonObject = JsonTools.getJsonObject("persons", persons);
        //    List<Person>相當於一個JSONArray物件
        JSONArray personsArray = (JSONArray)jsonObject.getJSONArray("persons");
        List<Person> persons2 = (List<Person>) personsArray.toCollection(personsArray, Person.class);
        System.out.println(persons2);
    }
{"persons":[{"address":"廣州","id":1,"name":"xiaoluo"},{"address":"上海","id":2,"name":"android"}]}
[Person [id=1, name=xiaoluo, address=廣州], Person [id=2, name=android, address=上海]]

③List<Map<String, String>>型別的json物件轉換

   @Test
    public void testMapJson()
    {
        List<Map<String, String>> list = new ArrayList<Map<String, String>>();
        Map<String, String> map1 = new HashMap<String, String>();
        map1.put("id", "001");
        map1.put("name", "xiaoluo");
        map1.put("age", "20");
        Map<String, String> map2 = new HashMap<String, String>();
        map2.put("id", "002");
        map2.put("name", "android");
        map2.put("age", "33");
        list.add(map1);
        list.add(map2);
        String listString = JsonTools.getJsonString("list", list);
        System.out.println(listString);
        
        JSONObject jsonObject = JsonTools.getJsonObject("list", list);
        JSONArray listArray = jsonObject.getJSONArray("list");
        List<Map<String, String>> list2 = (List<Map<String, String>>) listArray.toCollection(listArray, Map.class);
        System.out.println(list2);
    }
{"list":[{"id":"001","age":"20","name":"xiaoluo"},{"id":"002","age":"33","name":"android"}]}
[{id=001, name=xiaoluo, age=20}, {id=002, name=android, age=33}]

通過上面的例子,我們可以瞭解瞭如何通過json-lib這個解析庫來實現javabean、List、Map等資料和json資料的互相轉換

2.gson(http://code.google.com/p/google-gson/)

下面我們來看看Google提供的gson這個json解析庫,同樣我們需要去下載gson這個jar包,匯入到我們的專案中

使用gson,我們可以非常輕鬆的實現資料物件和json物件的相互轉化,其中我們最常用的就是兩個方法,一個是fromJSON(),將json物件轉換成我們需要的資料物件,另一個是toJSON(),這個就是將我們的資料物件轉換成json物件。下面我們也通過一個綜合的例子來看看gson的使用方法:

public class JsonService
{
    public Person getPerson()
    {
        Person person = new Person(1, "xiaoluo", "廣州");
        return person;
    }
    
    public List<Person> getPersons()
    {
        List<Person> persons = new ArrayList<Person>();
        Person person = new Person(1, "xiaoluo", "廣州");
        Person person2 = new Person(2, "android", "上海");
        persons.add(person);
        persons.add(person2);
        return persons;
    }
    
    public List<String> getString()
    {
        List<String> list = new ArrayList<String>();
        list.add("廣州");
        list.add("上海");
        list.add("北京");
        return list;
    }
    
    public List<Map<String, String>> getMapList()
    {
        List<Map<String, String>> list = new ArrayList<Map<String, String>>();
        Map<String, String> map1 = new HashMap<String, String>();
        map1.put("id", "001");
        map1.put("name", "xiaoluo");
        map1.put("age", "20");
        Map<String, String> map2 = new HashMap<String, String>();
        map2.put("id", "002");
        map2.put("name", "android");
        map2.put("age", "33");
        list.add(map1);
        list.add(map2);
        return list;
    }
}
public static void main(String[] args)
    {
        Gson gson = new Gson();
        JsonService jsonService = new JsonService();
        Person person = jsonService.getPerson();
        System.out.println("person: " + gson.toJson(person));
        //    對於Object型別,使用 fromJson(String, Class)方法來將Json物件轉換成Java物件
        Person person2 = gson.fromJson(gson.toJson(person), Person.class);
        System.out.println(person2);
        System.out.println("------------------------------------------------");
        
        List<Person> persons = jsonService.getPersons();
        System.out.println("persons: " + gson.toJson(persons));
        /*
         * 對於泛型物件,使用fromJson(String, Type)方法來將Json物件轉換成對應的泛型物件
         * new TypeToken<>(){}.getType()方法
         */
        List<Person> persons2 = gson.fromJson(gson.toJson(persons), new TypeToken<List<Person>>(){}.getType());
        System.out.println(persons2);
        System.out.println("------------------------------------------------");
        
        List<String> list = jsonService.getString();
        System.out.println("String---->" + gson.toJson(list));
        List<String> list2 = gson.fromJson(gson.toJson(list), new TypeToken<List<String>>(){}.getType());
        System.out.println("list2---->" + list2);
        System.out.println("------------------------------------------------");
        
        List<Map<String, String>> listMap = jsonService.getMapList();
        System.out.println("Map---->" + gson.toJson(listMap));
        List<Map<String, String>> listMap2 = gson.fromJson(gson.toJson(listMap), new TypeToken<List<Map<String, String>>>(){}.getType());
        System.out.println("listMap2---->" + listMap2);
        System.out.println("------------------------------------------------");
    }

看看控制檯的輸出:

person: {"id":1,"name":"xiaoluo","address":"廣州"}
Person [id=1, name=xiaoluo, address=廣州]
------------------------------------------------
persons: [{"id":1,"name":"xiaoluo","address":"廣州"},{"id":2,"name":"android","address":"上海"}]
[Person [id=1, name=xiaoluo, address=廣州], Person [id=2, name=android, address=上海]]
------------------------------------------------
String---->["廣州","上海","北京"]
list2---->[廣州, 上海, 北京]
------------------------------------------------
Map---->[{"id":"001","age":"20","name":"xiaoluo"},{"id":"002","age":"33","name":"android"}]
listMap2---->[{id=001, age=20, name=xiaoluo}, {id=002, age=33, name=android}]
------------------------------------------------

三、在Android客戶端解析伺服器端的json資料

下面我們來完成一個綜合的例子,Android客戶端通過一個AsyncTask非同步任務請求伺服器端的某些資料,然後在解析完這些資料後,將得到的資料內容更新到我們的Spinner這個UI控制元件當中。

我們首先來看下伺服器端的程式碼:

@WebServlet("/CityServlet")
public class CityServlet extends HttpServlet
{
    private static final long serialVersionUID = 1L;

    public CityServlet()
    {
        super();
    }

    protected void doGet(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException
    {
        this.doPost(request, response);
    }

    protected void doPost(HttpServletRequest request,
            HttpServletResponse response) throws ServletException, IOException
    {
        response.setContentType("text/html;charset=utf-8");
        request.setCharacterEncoding("utf-8");
        response.setCharacterEncoding("utf-8");
        PrintWriter writer = response.getWriter();
        
        String type = request.getParameter("type");
        if("json".equals(type))
        {
            List<String> cities = new ArrayList<String>();
            cities.add("廣州");
            cities.add("上海");
            cities.add("北京");
            cities.add("湖南");
            Map<String, List<String>> map = new HashMap<String, List<String>>();
            map.put("cities", cities);
            String citiesString = JSON.toJSONString(map);
            writer.println(citiesString);
        }
        
        writer.flush();
        writer.close();
    }

}

如果客戶端請求的引數是type=json,則響應給客戶端一個json資料格式

接著來看看客戶端的程式碼,首先看看客戶端的佈局檔案,其實就是一個按鈕和一個Spinner控制元件,當點選按鈕後,通過http協議請求伺服器端的資料,然後在接收到後再更新我們的Spinner控制元件的資料

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="64dp"
        android:layout_marginTop="64dp"
        android:textSize="20sp"
        android:text="城市" />
    
    <Spinner 
        android:id="@+id/spinner"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignTop="@id/textView1"
        android:layout_toRightOf="@id/textView1"/>

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignLeft="@+id/textView1"
        android:layout_below="@+id/spinner"
        android:layout_marginLeft="22dp"
        android:layout_marginTop="130dp"
        android:text="載入資料" />

</RelativeLayout>

在Android客戶端寫一個解析json資料格式的類:

public class JsonUtils
{
    /**
     * @param citiesString    從伺服器端得到的JSON字串資料
     * @return    解析JSON字串資料,放入List當中
     */
    public static List<String> parseCities(String citiesString)
    {
        List<String> cities = new ArrayList<String>();
        
        try
        {
            JSONObject jsonObject = new JSONObject(citiesString);
            JSONArray jsonArray = jsonObject.getJSONArray("cities");
            for(int i = 0; i < jsonArray.length(); i++)
            {
                cities.add(jsonArray.getString(i));
            }
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        
        return cities;
    }
}

當然我們的HttpUtils類也不可少:

public class HttpUtils
{
    /**
     * @param path    請求的伺服器URL地址
     * @param encode    編碼格式
     * @return    將伺服器端返回的資料轉換成String
     */
    public static String sendPostMessage(String path, String encode)
    {
        String result = "";
        HttpClient httpClient = new DefaultHttpClient();
        try
        {
            HttpPost httpPost = new HttpPost(path);
            HttpResponse httpResponse = httpClient.execute(httpPost);
            if(httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK)
            {
                HttpEntity httpEntity = httpResponse.getEntity();
                if(httpEntity != null)
                {
                    result = EntityUtils.toString(httpEntity, encode);
                }
            }
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        finally
        {
            httpClient.getConnectionManager().shutdown();
        }
        
        return result;
    }
}

最後來看看我們的MainActivity類:

public class MainActivity extends Activity
{
    private Spinner spinner;
    private Button button;
    private ArrayAdapter<String> adapter;
    private ProgressDialog dialog;
    private final String CITY_PATH_JSON = "http://172.25.152.34:8080/httptest/CityServlet?type=json";
    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        spinner = (Spinner)findViewById(R.id.spinner);
        button = (Button)findViewById(R.id.button);
        dialog = new ProgressDialog(MainActivity.this);
        button.setOnClickListener(new OnClickListener()
        {
            @Override
            public void onClick(View v)
            {
                dialog.setTitle("提示資訊");
                dialog.setMessage("loading......");
                dialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);
                dialog.setCancelable(false);
                
                new MyAsyncTask().execute(CITY_PATH_JSON);
            }
        });
    }

    public class MyAsyncTask extends AsyncTask<String, Void, List<String>>
    {
        @Override
        protected void onPreExecute()
        {
            dialog.show();
        }
        @Override
        protected List<String> doInBackground(String... params)
        {
            List<String> cities = new ArrayList<String>();
            String citiesString = HttpUtils.sendPostMessage(params[0], "utf-8");
            //    解析伺服器端的json資料
            cities = JsonUtils.parseCities(citiesString);return cities;
        }
        @Override
        protected void onPostExecute(List<String> result)
        {
            adapter = new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_spinner_item, result);
            adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
            spinner.setAdapter(adapter);
            dialog.dismiss();
        }
    }
    
    @Override
    public boolean onCreateOptionsMenu(Menu menu)
    {
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

}

當然別往了開啟我們的網路授權

<uses-permission android:name="android.permission.INTERNET"/>

最後我們來看看效果圖:

這樣我們就完成了客戶端與伺服器端通過json來進行資料的交換

 

總結:本篇隨筆主要講解了JSON這種輕量級的資料交換格式的概念,以及講解了兩種解析json資料的解析類(json-lib以及gson),最後通過一個小例子實現了在Android客戶端和伺服器端使用json這種資料格式來進行資料的交換。

相關文章