深刻了解Struts2返回JSON數據的原理及具體應用範例

早在我剛學Struts2之初的時候,就想寫一篇文章來闡述Struts2如何返回JSON數據的原理和具體應用了,但苦於一直忙於工做難以抽身,漸漸的也淡忘了此事。直到前兩天有同事在工做中遇到這個問題,來找我詢問,我又細細地給他講了一遍以後,才以爲不管如何要抽一個小時的時間來寫這篇文章,從頭至尾將Struts2與JSON的關係說清楚。 html

 

其實網絡中,關於這個問題的答案已經是海量,我當初也是從這海量的答案中吸取精華,纔將「Struts2返回JSON數據」這個問題搞清楚的。可是這些海量的答案,有一個共同的缺陷,就是做者們只關注問題核心,即「如何在具體的Struts2應用中返回JSON數據到客戶端」如何實現,而對於"爲什麼要這樣實現"以及實現的本質卻解釋的不甚了了,在筆者看來這只是「授人以魚」而非筆者所推崇的「授人以魚的同時,授人以漁」。在這篇文章中,筆者將總結前輩們的經驗,並結合本身的理解,來從理論到實踐由淺入深的說明「Struts2返回JSON數據」這一問題。 java

 

JSON(JavaScript Object Notation) apache

 

首先來看一下JSON官方對於「JSON」的解釋: json

JSON(JavaScript Object Notation) 是一種輕量級的數據交換格式。易於人閱讀和編寫。同時也易於機器解析和生成。它基於JavaScript Programming Language, Standard ECMA-262 3rd Edition - December 1999的一個子集。 JSON採用徹底獨立於語言的文本格式,可是也使用了相似於C語言家族的習慣(包括C, C++, C#, Java, JavaScript, Perl, Python等)。這些特性使JSON成爲理想的數據交換語言。(更多內容請參見JSON官網http://json.org/json-zh.html) 數組


JSON建構於兩種結構: 瀏覽器

「名稱/值」對的集合(A collection of name/value pairs)。不一樣的語言中,它被理解爲對象(object),紀錄(record),結構(struct),字典(dictionary),哈希表(hash table),有鍵列表(keyed list),或者關聯數組 (associative array)。 服務器


值的有序列表(An ordered list of values)。在大部分語言中,它被理解爲數組(array)。 網絡

由於JSON中的值(value)能夠是雙引號括起來的字符串(string)、數值(number)、true、false、 null、對象(object)或者數組(array),且這些結構能夠嵌套,這種特性給予JSON表達數據以無限的可能:它既能夠表達一個簡單的key/value,也能夠表達一個複雜的Map或List,並且它是易於閱讀和理解的。 jsp

 

 

Struts2中JSON的用武之地 post


由於JSON是脫離語言的理想的數據交換格式,因此它被頻繁的應用在客戶端與服務器的通訊過程當中,這一點是毋庸置疑的。而在客戶端與服務器的通訊過程當中,JSON數據的傳遞又被分爲服務器向客戶端傳送JSON數據,和客戶端向服務器傳送JSON數據,前者的核心過程當中將對象轉換成JSON,然後者的核心是將JSON轉換成對象,這是本質的區別。另外,值得一提的是,JSON數據在傳遞過程當中,其實就是傳遞一個普通的符合JSON語法格式的字符串而已,所謂的「JSON對象」是指對這個JSON字符串解析和包裝後的結果,這一點請牢記,由於下面的內容會依賴這一點。

 

Struts2返回JSON數據到客戶端


這是最多見的需求,在AJAX大行其道的今天,向服務器請求JSON數據已成爲每個WEB應用必備的功能。拋開Struts2暫且不提,在常規WEB應用中由服務器返回JSON數據到客戶端有兩種方式:一是在Servlet中輸出JSON串,二是在JSP頁面中輸出JSON串。上文提到,服務器像客戶端返回JSON數據,其實就是返回一個符合JSON語法規範的字符串,因此在上述兩種 方法中存在一個共同點,就是將須要返回的數據包裝稱符合JSON語法規範的字符串後在頁面中顯示。以下所示

 

使用Servlet返回JSON數據到客戶端:

 

Java代碼   收藏代碼
  1. package cn.ysh.studio.struts2.json.demo.servlet;  
  2.   
  3. import java.io.IOException;  
  4. import java.io.PrintWriter;  
  5.   
  6. import javax.servlet.ServletException;  
  7. import javax.servlet.http.HttpServlet;  
  8. import javax.servlet.http.HttpServletRequest;  
  9. import javax.servlet.http.HttpServletResponse;  
  10.   
  11. import net.sf.json.JSONObject;  
  12.   
  13. import cn.ysh.studio.struts2.json.demo.bean.User;  
  14.   
  15. public class JSON extends HttpServlet {  
  16.   
  17.     /** 
  18.      *  
  19.      */  
  20.     private static final long serialVersionUID = 1L;  
  21.   
  22.     /** 
  23.      * The doGet method of the servlet. <br> 
  24.      * 
  25.      * This method is called when a form has its tag value method equals to get. 
  26.      *  
  27.      * @param request the request send by the client to the server 
  28.      * @param response the response send by the server to the client 
  29.      * @throws ServletException if an error occurred 
  30.      * @throws IOException if an error occurred 
  31.      */  
  32.     public void doGet(HttpServletRequest request, HttpServletResponse response)  
  33.             throws ServletException, IOException {  
  34.   
  35.         response.setContentType("text/html");  
  36.         PrintWriter out = response.getWriter();  
  37.         //將要被返回到客戶端的對象  
  38.         User user=new User();  
  39.         user.setId("123");  
  40.         user.setName("JSONServlet");  
  41.         user.setPassword("JSON");  
  42.         user.setSay("Hello , i am a servlet !");  
  43.         JSONObject json=new JSONObject();  
  44.         json.accumulate("success"true);  
  45.         json.accumulate("user", user);  
  46.         out.println(json.toString());  
  47. //      由於JSON數據在傳遞過程當中是以普通字符串形式傳遞的,因此咱們也能夠手動拼接符合JSON語法規範的字符串輸出到客戶端  
  48. //      如下這兩句的做用與38-46行代碼的做用是同樣的,將向客戶端返回一個User對象,和一個success字段  
  49. //      String jsonString="{\"user\":{\"id\":\"123\",\"name\":\"JSONServlet\",\"say\":\"Hello , i am a servlet !\",\"password\":\"JSON\"},\"success\":true}";  
  50. //      out.println(jsonString);  
  51.         out.flush();  
  52.         out.close();  
  53.     }  
  54.   
  55.     /** 
  56.      * The doPost method of the servlet. <br> 
  57.      * 
  58.      * This method is called when a form has its tag value method equals to post. 
  59.      *  
  60.      * @param request the request send by the client to the server 
  61.      * @param response the response send by the server to the client 
  62.      * @throws ServletException if an error occurred 
  63.      * @throws IOException if an error occurred 
  64.      */  
  65.     public void doPost(HttpServletRequest request, HttpServletResponse response)  
  66.             throws ServletException, IOException {  
  67.         doGet(request, response);  
  68.     }  
  69.   
  70. }  

 

結果在乎料之中,以下圖所示:

 

 

 

 

使用JSP(或html等)返回JSON數據到客戶端:

 

Java代碼   收藏代碼
  1. <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>  
  2. {"user":{"id":"123","name":"JSONJSP","say":"Hello , i am a JSP !","password":"JSON"},"success":true}  

 

這個就不用去看結果了吧。

 

再回到Struts,在Struts的MVC模型中,Action替代Servlet擔當了Model的角色,因此對於Struts而言,返回JSON數據到客戶端,跟傳統的WEB應用同樣,存在兩種方式,即在Action中輸出JSON數據,和在視圖資源中輸出JSON數據。再往下細分的話,在Action中輸出JSON數據又分爲兩種方式,一是使用傳統方式輸出本身包裝後的JSON數據,二是使用Struts自帶的JSON數據封裝功能來自動包裝並返回JSON數據。

 

在視圖資源中輸出JSON數據


Action處理完用戶請求後,將數據存放在某一位置,如request中,並返回視圖,而後Struts將跳轉至該視圖資源,在該視圖中,咱們須要作的是將數據從存放位置中取出,而後將其轉換爲JSON字符串,輸出在視圖中。這跟傳統WEB應用中在JSP頁面輸出JSON數據的作法一模一樣:

 

Java代碼   收藏代碼
  1. public String testByJSP() {  
  2.         User user = new User();  
  3.         user.setId("123");  
  4.         user.setName("Struts2");  
  5.         user.setPassword("123");  
  6.         user.setSay("Hello world !");  
  7.         JSONObject jsonObject=new JSONObject();  
  8.         jsonObject.accumulate("user", user);  
  9.         //這裏在request對象中放了一個data,因此struts的result配置中不能有type="redirect"  
  10.         ServletActionContext.getRequest().setAttribute("data", jsonObject.toString());  
  11.         return SUCCESS;  
  12.     };  

 

由於是常規的Struts流程配置,因此配置內容就再也不展現了。

 

JSP代碼就很是簡單了,

 

Java代碼   收藏代碼
  1. <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>  
  2. ${data }  

 

結果如圖所示:

 

 

 

 

在Action中以傳統方式輸出JSON數據


這一點跟傳統的Servlet的處理方式基本上如出一轍,代碼以下

 

Java代碼   收藏代碼
  1. public void doAction() throws IOException{  
  2.         HttpServletResponse response=ServletActionContext.getResponse();  
  3.         //如下代碼從JSON.java中拷過來的  
  4.         response.setContentType("text/html");  
  5.         PrintWriter out;  
  6.         out = response.getWriter();  
  7.         //將要被返回到客戶端的對象  
  8.         User user=new User();  
  9.         user.setId("123");  
  10.         user.setName("JSONActionGeneral");  
  11.         user.setPassword("JSON");  
  12.         user.setSay("Hello , i am a action to print a json!");  
  13.         JSONObject json=new JSONObject();  
  14.         json.accumulate("success"true);  
  15.         json.accumulate("user", user);  
  16.         out.println(json.toString());  
  17. //      由於JSON數據在傳遞過程當中是以普通字符串形式傳遞的,因此咱們也能夠手動拼接符合JSON語法規範的字符串輸出到客戶端  
  18. //      如下這兩句的做用與38-46行代碼的做用是同樣的,將向客戶端返回一個User對象,和一個success字段  
  19. //      String jsonString="{\"user\":{\"id\":\"123\",\"name\":\"JSONActionGeneral\",\"say\":\"Hello , i am a action to print a json!\",\"password\":\"JSON\"},\"success\":true}";  
  20. //      out.println(jsonString);  
  21.         out.flush();  
  22.         out.close();  
  23.     }  

 

struts.xml中的配置:

 

Java代碼   收藏代碼
  1. <package name="default" extends="struts-default" namespace="/">  
  2.     <action name="testJSONFromActionByGeneral" class="cn.ysh.studio.struts2.json.demo.action.UserAction" method="doAction">  
  3.     </action>  
  4. </package>  

 

注意:這個action沒有result,且doAction方法沒有返回值!

 

就再也不貼圖了,由於結果可想而知!

 

 

在Action中以Struts2的方式輸出JSON數據

本着「不重複發明輪子」的原則,咱們將轉換JSON數據的工做交給Struts2來作,那麼相對於在Action中以傳統方式輸出JSON不一樣的是,Action是須要將注意力放在業務處理上,而無需關心處理結果是如何被轉換成JSON被返回客戶端的——這些 工做經過簡單的配置,Struts2會幫咱們作的更好。

 

Java代碼   收藏代碼
  1. public String testByAction() {  
  2.         // dataMap中的數據將會被Struts2轉換成JSON字符串,因此這裏要先清空其中的數據  
  3.         dataMap.clear();  
  4.         User user = new User();  
  5.         user.setId("123");  
  6.         user.setName("JSONActionStruts2");  
  7.         user.setPassword("123");  
  8.         user.setSay("Hello world !");  
  9.         dataMap.put("user", user);  
  10.         // 放入一個是否操做成功的標識  
  11.         dataMap.put("success"true);  
  12.         // 返回結果  
  13.         return SUCCESS;  
  14.     }  

 

 

struts.xml中action的配置:

 

Java代碼   收藏代碼
  1. <package name="json" extends="json-default" namespace="/test">  
  2.         <action name="testByAction"  
  3.             class="cn.ysh.studio.struts2.json.demo.action.UserAction" method="testByAction">  
  4.             <result type="json">  
  5.                 <!-- 這裏指定將被Struts2序列化的屬性,該屬性在action中必須有對應的getter方法 -->  
  6.                 <param name="root">dataMap</param>  
  7.             </result>  
  8.         </action>  
  9. </package>  

 

凡是使用Struts2序列化對象到JSON的action,所在的package必須繼承自json-default,注意,這裏惟一的result,沒有指定name屬性。

 

結果以下圖所示:

 

 

 

上面很詳細的說明了在WEB應用中如何返回JSON數據到客戶端,講了那麼多種方式,涉及的技術核心無非只有兩點:


一、將對象轉換成符合JSON語法格式的字符串;
二、將符合JSON語法格式的字符串返回客戶端;


第二點是整個實現過程的本質,但卻不難作到;第一點其實也不難,他甚至有兩種作法,一是經過字符串拼接方式,而是經過JSONObject以對象方式轉換。看下面的一個例子:

 

Java代碼   收藏代碼
  1. package cn.ysh.studio.struts2.json.demo.test;  
  2.   
  3. import cn.ysh.studio.struts2.json.demo.bean.User;  
  4. import net.sf.json.JSONObject;  
  5.   
  6. public class JSONTest {  
  7.   
  8.     /** 
  9.      * 將普通的pojo轉換成JSON字符串 
  10.      * @return  
  11.      */  
  12.     public JSONObject bean2json() {  
  13.         User user = new User();  
  14.         user.setId("JSONTest");  
  15.         user.setName("JSONTest");  
  16.         user.setPassword("JSON");  
  17.         user.setSay("Hello,i am JSONTest.java");  
  18.         JSONObject jsonObject = new JSONObject();  
  19.         jsonObject.accumulate("user", user);  
  20.         System.out.println("User轉換後的字符串:"+jsonObject.toString());  
  21.         return jsonObject;  
  22.     }  
  23.   
  24.     /** 
  25.      * 從JSONObject對象中反向解析出User對象 
  26.      * @param jsonObject 
  27.      */  
  28.     public void json2bean(JSONObject jsonObject) {  
  29.         User user=(User)JSONObject.toBean((JSONObject)jsonObject.get("user"),User.class);  
  30.         System.out.println("轉換獲得的User對象的Name爲:"+user.getName());  
  31.     }  
  32.   
  33.     public static void main(String[] s) {  
  34.         JSONTest tester=new JSONTest();  
  35.         tester.json2bean(tester.bean2json());  
  36.     }  
  37. }  

 

 

JSON格式的字符串返回到客戶端後,客戶端會將其解析並封裝成真正的JSON對象,以供JS調用。

 


總結上述,其實只要明白了服務器返回JSON數據到客戶端的原理,作起來就遊刃有餘了,他甚至有很是多的可選方案,但既然是基於Struts2的實現,那麼確定仍是要用Struts2的方式來作啦,由於這樣確實能夠省不少事。另外,在文章的最後,說明一下返回JSON數據時在result中配置的參數的含義及其常見常見配置吧:

 

Java代碼   收藏代碼
  1. <result type="json">  
  2.                 <!-- 這裏指定將被Struts2序列化的屬性,該屬性在action中必須有對應的getter方法 -->  
  3.                 <!-- 默認將會序列全部有返回值的getter方法的值,而不管該方法是否有對應屬性 -->  
  4.                 <param name="root">dataMap</param>  
  5.                 <!-- 指定是否序列化空的屬性 -->  
  6.                 <param name="excludeNullProperties">true</param>  
  7.                 <!-- 這裏指定將序列化dataMap中的那些屬性 -->  
  8.                 <param name="includeProperties">  
  9.                     userList.*  
  10.                 </param>  
  11.                 <!-- 這裏指定將要從dataMap中排除那些屬性,這些排除的屬性將不被序列化,一半不與上邊的參數配置同時出現 -->  
  12.                 <param name="excludeProperties">  
  13.                     SUCCESS  
  14.                 </param>  
  15. </result>  

 

值得一提的是經過Struts2來返回JSON數據,在IE中會提示下載,這個不用關心,換個瀏覽器就能正常展現JSON數據,而在JS調用中,更是毫無影響。

 

下面是整個Action的完整代碼:

 

Java代碼   收藏代碼
  1. package cn.ysh.studio.struts2.json.demo.action;  
  2.   
  3. import java.io.IOException;  
  4. import java.io.PrintWriter;  
  5. import java.util.HashMap;  
  6. import java.util.Map;  
  7.   
  8. import javax.servlet.http.HttpServletResponse;  
  9.   
  10. import org.apache.struts2.ServletActionContext;  
  11.   
  12. import net.sf.json.JSONObject;  
  13.   
  14. import cn.ysh.studio.struts2.json.demo.bean.User;  
  15.   
  16. import com.opensymphony.xwork2.ActionSupport;  
  17.   
  18. public class UserAction extends ActionSupport {  
  19.   
  20.     /** 
  21.      *  
  22.      */  
  23.     private static final long serialVersionUID = 1L;  
  24.   
  25.     //將會被Struts2序列化爲JSON字符串的對象  
  26.     private Map<String, Object> dataMap;  
  27.   
  28.     /** 
  29.      * 構造方法 
  30.      */  
  31.     public UserAction() {  
  32.         //初始化Map對象  
  33.         dataMap = new HashMap<String, Object>();  
  34.     }  
  35.   
  36.     /** 
  37.      * 測試經過action以視圖方式返回JSON數據 
  38.      * @return  
  39.      */  
  40.     public String testByJSP() {  
  41.         User user = new User();  
  42.         user.setId("123");  
  43.         user.setName("JSONActionJSP");  
  44.         user.setPassword("123");  
  45.         user.setSay("Hello world !");  
  46.         JSONObject jsonObject=new JSONObject();  
  47.         jsonObject.accumulate("user", user);  
  48.         jsonObject.accumulate("success"true);  
  49.         //這裏在request對象中放了一個data,因此struts的result配置中不能有type="redirect"  
  50.         ServletActionContext.getRequest().setAttribute("data", jsonObject.toString());  
  51.         return SUCCESS;  
  52.     };  
  53.   
  54.     /** 
  55.      * 測試經過action以Struts2默認方式返回JSON數據 
  56.      * @return  
  57.      */  
  58.     public String testByAction() {  
  59.         // dataMap中的數據將會被Struts2轉換成JSON字符串,因此這裏要先清空其中的數據  
  60.         dataMap.clear();  
  61.         User user = new User();  
  62.         user.setId("123");  
  63.         user.setName("JSONActionStruts2");  
  64.         user.setPassword("123");  
  65.         user.setSay("Hello world !");  
  66.         dataMap.put("user", user);  
  67.         // 放入一個是否操做成功的標識  
  68.         dataMap.put("success"true);  
  69.         // 返回結果  
  70.         return SUCCESS;  
  71.     }  
  72.   
  73.     /** 
  74.      * 經過action是以傳統方式返回JSON數據 
  75.      * @throws IOException 
  76.      */  
  77.     public void doAction() throws IOException{  
  78.         HttpServletResponse response=ServletActionContext.getResponse();  
  79.         //如下代碼從JSON.java中拷過來的  
  80.         response.setContentType("text/html");  
  81.         PrintWriter out;  
  82.         out = response.getWriter();  
  83.         //將要被返回到客戶端的對象  
  84.         User user=new User();  
  85.         user.setId("123");  
  86.         user.setName("JSONActionGeneral");  
  87.         user.setPassword("JSON");  
  88.         user.setSay("Hello , i am a action to print a json!");  
  89.         JSONObject json=new JSONObject();  
  90.         json.accumulate("success"true);  
  91.         json.accumulate("user", user);  
  92.         out.println(json.toString());  
  93. //      由於JSON數據在傳遞過程當中是以普通字符串形式傳遞的,因此咱們也能夠手動拼接符合JSON語法規範的字符串輸出到客戶端  
  94. //      如下這兩句的做用與38-46行代碼的做用是同樣的,將向客戶端返回一個User對象,和一個success字段  
  95. //      String jsonString="{\"user\":{\"id\":\"123\",\"name\":\"JSONActionGeneral\",\"say\":\"Hello , i am a action to print a json!\",\"password\":\"JSON\"},\"success\":true}";  
  96. //      out.println(jsonString);  
  97.         out.flush();  
  98.         out.close();  
  99.     }  
  100.       
  101.     /** 
  102.      * Struts2序列化指定屬性時,必須有該屬性的getter方法,實際上,若是沒有屬性,而只有getter方法也是能夠的 
  103.      * @return  
  104.      */  
  105.     public Map<String, Object> getDataMap() {  
  106.         return dataMap;  
  107.     }  
  108.   
  109. }  

 

完整的struts.xml配置文件以下:

 

Xml代碼   收藏代碼
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"   
  3.     "http://struts.apache.org/dtds/struts-2.0.dtd">  
  4. <struts>  
  5.     <package name="json" extends="json-default" namespace="/test">  
  6.         <action name="testByAction"  
  7.             class="cn.ysh.studio.struts2.json.demo.action.UserAction" method="testByAction">  
  8.             <result type="json">  
  9.                 <!-- 這裏指定將被Struts2序列化的屬性,該屬性在action中必須有對應的getter方法 -->  
  10.                 <!-- 默認將會序列全部有返回值的getter方法的值,而不管該方法是否有對應屬性 -->  
  11.                 <param name="root">dataMap</param>  
  12.                 <!-- 指定是否序列化空的屬性 -->  
  13.                 <!-- 
  14.                 <param name="excludeNullProperties">true</param> 
  15.                 -->  
  16.                 <!-- 這裏指定將序列化dataMap中的那些屬性 -->  
  17.                 <!--   
  18.                 <param name="includeProperties">  
  19.                     userList.*  
  20.                 </param>  
  21.                  -->  
  22.                 <!-- 這裏指定將要從dataMap中排除那些屬性,這些排除的屬性將不被序列化,一半不與上邊的參數配置同時出現 -->  
  23.                 <!--   
  24.                 <param name="excludeProperties">  
  25.                     SUCCESS  
  26.                 </param>  
  27.                 -->  
  28.             </result>  
  29.         </action>  
  30.     </package>  
  31.     <package name="default" extends="struts-default" namespace="/">  
  32.         <action name="testJSONFromActionByGeneral"  
  33.             class="cn.ysh.studio.struts2.json.demo.action.UserAction" method="doAction">  
  34.         </action>  
  35.         <action name="testByJSP"  
  36.             class="cn.ysh.studio.struts2.json.demo.action.UserAction" method="testByJSP">  
  37.             <result name="success">/actionJSP.jsp</result>  
  38.         </action>  
  39.     </package>  
  40. </struts>  

 

 

最後,附上整個範例工程(一個MyEclipse工程)源碼。

相關文章
相關標籤/搜索