<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    菠蘿三國

    大江東去,浪淘盡...
    隨筆 - 34, 文章 - 47, 評論 - 22, 引用 - 0
    數據加載中……

    Web Service深度編程——Axis序列化/反序列化器開發指南

     

    JAVA——咖啡館Velocity 的應用示例- -| 回首頁 | 2005年索引 | - -Web Service 的開發層次
    Web Service深度編程——Axis序列化/反序列化器- -
    Tag: Web   Service   Axis                            

    Web Service深度編程——Axis序列化/反序列化器開發指南


    作者:薛谷雨

    作者簡介

    薛谷雨是NORDSAN(北京)信息科技開發有限公司高級JAVA研發工程師,正致力于企業級異構數據交換的服務器產品的研發,在J2EE和WEB SERVICE方面有較為豐富的開發經驗,你可以通過 rainight@126.com 與他取得聯系。

    前言

    Axis是Apache組織推出的SOAP引擎,Axis項目是Apache組織著名的SOAP項目的后繼項目。Axis為開發者提供了大量的序列化/反序列化器,能夠基本滿足大部分應用。但在某些情況下,對特定的對象,現有的序列化/反序列化器不能勝任,于是只有開發人員自己實現專用于此對象的序列化/反序列化器插入到Axis中來完成序列化工作。考慮到Web Service是一門新興技術,中文資料大多是泛泛的講解,關于序列化/反序列化器的開發鮮有較為深入的介紹,本文提供一份較為完整的開發指南,并提供了一個十分有用的實現,即序列化JDOM模型的Element,使其可以通過Web 服務在網絡上傳輸,我想這一擴展是許多采用JDOM作為XML解析工具的開發人員都夢寐以求的功能。通過本文的介紹和實例,希望能起到拋磚引玉的作用,讀者在閱讀完本文之后可以輕松的實現針對于任何非符合BEAN規范的對象的序列化/反序列化器。

    本文所面對的讀者需要有一定的使用Axis做Web服務開發的開發經驗,因此關于如何Axis的基礎知識并不在本文的介紹范圍,如果讀者對此感興趣,可以參考本文最后的參考資料部分,去相應的網站進行學習。

    序列化/反序列化器簡介

    序列化/反序列化器在英文中的對應翻譯是Serializer/Deserializer,一個序列化器的功能是遵循一定的映射規則和編碼風格,將一種類型的JAVA對象通過某種特定的機制,轉換成為XML描述的形式;反序列化器的功能是序列化器所做工作的逆操作,兩者相輔相成,成對出現。Axis中的序列化/反序列化器采用設計范式中的工廠模式,每一個Serializer唯一對應一個SerializerFactory;每一個Deserializer唯一對應一個DeserializerFactory。一種類型的JAVA對象具體要采用哪個序列化/反序列化器需要在提供Web服務的服務器和調用Web服務的客戶端分別配置,關于這一部分如何配置,我將在本文后面的內容中進行詳細介紹。Axis已經為開發者提供了豐富的序列化/反序列化器,對于java的基本數據類型,絕大部分常用的容器類(比如數組類型,Vector類型等)都提供了實現,特別是提供了對W3C的DOM對象(比如Document, Element等)和符合Bean規范的JAVA對象提供了功能完善的序列化/反序列化器,因此我們在需要的時候只要在配置文件中配置一下就可以直接使用。如果對象中包含其它類型的對象,比如Vector中包含一組Bean對象,Axis會自動疊代的調用序列化器,最終拼裝成唯一的XML表述。在還原成JAVA對象時,也遵循這樣的疊代操作逆向進行。關于Axis到底內置了哪些序列化/反序列化器,您可以參照Axis的API文檔中包org.apache.axis.encoding.ser下的類的名稱"望文生義"的了解一下,在以后的開發中做到心中有數。但對于一些特殊類型的對象(其實我們自己開發的很大一部分類都是這種特殊類型的對象,很少有絕對符合Bean規范的),需要通過Web服務進行傳遞,我們不得不開發自己的序列化/反序列化器。

    開發篇

    開發自己的序列化/反序列化器是一個激動人心的工作,但是卻并不復雜,需要做的事情包括實現名成為org.apache.axis.encoding的包中的SerializerFactory,Serializer,DeserializerFactory和Deserializer這四個接口。下面我將結合一個實例來講解序列化/反序列化器的開發方法,希望讀者能夠一邊參看本文提供的源代碼一邊學習。

    JDOM作為一款比較"另類"的XML解析工具(因為它不符合W3C的DOM模型,自己另立一套)默默地占領著java世界里的xml解析器的半壁江山,由于其簡潔的設計和方便靈活的API調用,已經漸漸成為了許多開發人員在進行XML開發的首選。但是Axis是建立在W3C的DOM模型的基礎之上,師出名們正派,自然不屑與JDOM為伍。因此當開發人員想將自己已經寫好的基于JDOM的應用模塊采用Web服務的方式發布的時候,不可避免的會遇到如何將JDOM模型下的對象如Document, Element等序列化的問題。在軟件工程師不會自己擴展Axis的序列化/反序列化器的時候,我們只能有兩個辦法達到這個目的,第一個就是更改以前應用模塊內的API設計,使暴露的入口參數和返回值參數都是W3C的對象類型,但這種做法并不現實,因為這一應用模塊往往不是獨立存在,牽一發將動全身,導致舊有系統架構的崩塌;另一種做法就是為這個模塊做一個代理類,它做的工作就對外接收或返回DOM模型的對象,對內轉換成JDOM模型的對象,然后轉發給應用模塊,繁瑣且效率低下。當我們向Axis注入了針對于JDOM模型的序列化/反序列化器后,這一工作便可以由Axis代勞了。下面我們將逐個開發這四個類:

    JDomElementSerializerFactory

    JDomElementSerializerFactory是一個工廠類,需要通過某種機制注冊到Axis引擎(具體方法見下面"服務器端應用篇");Axis通過調用它,來實例化JDomElementSerializer。Axis 提供了BaseSerializerFactory,這個類是一個抽象類,并實現其中包含了一些可重用的代碼。我們自己開發的工廠類只需簡單繼承這個類就可以。構造函數中需要調用父類的構造函數將序列器類下面是它的源代碼:

    package org.apache.axis.encoding.ser;

    public class JDomElementSerializerFactory
    extends BaseSerializerFactory {

    public JDomElementSerializerFactory() {
      super(JDomElementSerializer.class);
    }
    }

    JDomElementSerializer

    JDomElementSerializer實現org.apache.axis.encoding.Serializer接口,其核心API是serialize(),我們需要在這個方法的內部完成對JDOM模型的Element的序列化工作,序列化的結果要保存在入口參數傳入的序列化上下文對象(SerializationContext)中:

    public void serialize(QName name, Attributes attributes, Object value,
    SerializationContext context) throws java.io.IOException {

      if (!(value instanceof Element))
        throw new IOException(
        Messages.getMessage("cant Serialize Object"));

      //獲取符合JDOM的Element對象
      Element root=(Element)value;

      //輸出到StringWriter
      XMLOutputter outputter=new XMLOutputter();//創建一個JDOM的XML輸出器
      StringWriter sw=new StringWriter();
      outputter.output(root,sw);

      //用支持W3C的DOM模型的Xerces解析器解析文本流
      DOMParser parser=new DOMParser();//創建一個DOM的XML解析器
      try {
        parser.parse(new org.xml.sax.InputSource(
            new java.io.StringReader(sw.toString())));
      }catch (Exception ex) {
        throw new java.io.IOException("序列化時產生錯誤");
      }

      //獲取符合DOM模型的Element對象
      org.w3c.dom.Element w3c_root =
          parser.getDocument().getDocumentElement();

      //放入序列化上下文對象中
      context.startElement(name, attributes);
      context.writeDOMElement(w3c_root);
      context.endElement();
    }

    JDomElementDeserializerFactory

    反序列化器的工廠類同序列化器的工廠類一樣的設計,在此不在贅述。代碼:

    package org.apache.axis.encoding.ser;

    public class JDomElementDeserializerFactory
    extends BaseDeserializerFactory {

    public JDomElementDeserializerFactory() {
      super(JDomElementDeserializer.class);
    }

    }

    JDomElementDeserializer

    用過SAX解析XML的讀者,對反序列化的實現比較容易理解,反序列化也采用了消息觸發的機制,我們只需繼承org.apache.axis.encoding.DeserializerImpl類,并覆蓋其中的onEndElement方法:

    /**
      * 在元素結束觸發反序列化的方法
      * @param namespace String 命名空間
      * @param localName String 本地名稱
      * @param context DeserializationContext 反序列化上下文
      * @throws SAXException
      */
    public void onEndElement(String namespace, String localName,
      DeserializationContext context) throws SAXException {

      try {
        //從反序列化上下文對象中獲取原始的消息元素
        MessageElement msgElem = context.getCurElement();
        if (msgElem != null) {
        MessageContext messageContext = context.getMessageContext();
        Boolean currentElement = (Boolean) messageContext.getProperty(
            DESERIALIZE_CURRENT_ELEMENT);

        //如果當前的消息元素本身需要反序列化
        if (currentElement != null && currentElement.booleanValue()) {
          org.w3c.dom.Element element = msgElem.getAsDOM();
          org.jdom.input.DOMBuilder db=new org.jdom.input.DOMBuilder();
          value=db.build(element);
          messageContext.setProperty(DESERIALIZE_CURRENT_ELEMENT,
            Boolean.FALSE);
          return;
        }

        //反序列化消息元素中的消息體
        java.util.ArrayList children = msgElem.getChildren();
        if (children != null) {

          //取得消息體
          msgElem = (MessageElement) children.get(0);
          if (msgElem != null) {
            org.w3c.dom.Element ret = msgElem.getAsDOM();
            org.jdom.input.DOMBuilder db=new org.jdom.input.DOMBuilder();

            //用DOMBuilder將DOM模型的Element,轉換成JDOM模型的Element
            value=db.build(ret);
          }
        }

        }
      }
      catch (Exception ex) {
        //錯誤,則記日志,并拋SAXException
        log.error(Messages.getMessage("exception00"), ex);
        throw new SAXException(ex);
      }
    }

    完成這四個類的編碼,序列化/反序列化器的開發工作基本完成,下面將詳細講解使用及部署方法。

    服務器端應用篇

    為了簡單起見,我們將一個很簡單的類通過Web服務發布,類中只有一個名稱為hello函數,函數的返回值為JDOM模型的Element。代碼如下:

    package test;

    import org.jdom.*;
    import java.rmi.RemoteException;

    public class Sample1 implements java.rmi.Remote{
    public Sample1() {
    }

    public Element hello(String name){
      Element root=new Element("root");
      Element hello=new Element("hello");
      hello.setText("hello,"+name+"!");
      root.addContent(hello);
      return root;
    }

    }

    關于如何將一個類發布成Web服務,在此并不進行介紹,相信讀者可以自己完成,我們只關注如何將序列化/反序列化器加入到我們的Web服務中。打開web服務的配置文件server-config.xml,編輯關于Sample1的服務的配置部分:



    注意上面代碼中的粗體字部分,是我們現在要添加的,它表述了如何將序列化反序列化器部署到Web服務中。

    部署到Web Server

    解壓縮本文后面附帶的源代碼,根目錄下有build.xml文件,讀者需要正確安裝配置好Apache Ant,然后運行

    Ant make

    編譯后可生成壓縮文件sample.war。將生成的war包部署到Tomcat4.1下,啟動Tomcat,本文默認的Tomcat監聽的http端口為8080。后面的客戶端測試程序也將通過連接這一端口訪問此Web服務。如果讀者的Tomcat不在8080端口上工作,那么客戶端程序也要進行相應的修改。最后啟動Tomcat,這部分操作完成。

    客戶端應用篇

    下面我們將編寫客戶端程序訪問剛才部署的Web服務,講解如何把我們編寫的序列化/反序列化器加載到客戶端應用程序中,下面是客戶端調用的代碼,注意斜體字部分,是關于序列化/反序列化器的注冊過程(如果你的Web服務器不是工作在8080端口,或采用了其他Web服務名,請自行更改下面程序中的url變量中的值),我們在test包下創建了一個名稱為 Client的類,代碼如下:

    package test;

    import org.apache.axis.client.Service;
    import org.apache.axis.client.Call;
    import org.apache.axis.utils.Options;
    import javax.xml.namespace.QName;

    public class Client {
    public Client() {
    }

    public static void main(String[] args) throws Exception{
      if(args.length<1){
        System.out.println("錯誤:缺少參數");
        System.exit(0);
    }
      //Web服務的URL
      String url="http://localhost:8080/sample/services/Sample1";
      Service service=new Service();
      Call call = (Call)service.createCall();
      call.setTargetEndpointAddress(url);

      //注冊序列化/反序列化器
      call.registerTypeMapping(org.jdom.Element.class,
          new QName("http://jdom.org","Element"),
        new org.apache.axis.encoding.ser.JDomElementSerializerFactory(),
        new org.apache.axis.encoding.ser.JDomElementDeserializerFactory());

      //設置調用方法
      call.setOperationName(
          new javax.xml.namespace.QName("http://test", "hello"));

      //Web服務調用
      java.lang.Object _resp = call.invoke(new java.lang.Object[] {args[0]});

      //輸出到屏幕
      org.jdom.output.XMLOutputter out=new org.jdom.output.XMLOutputter();
      out.output( (org.jdom.Element) _resp, System.out);

    }

    }

    編譯后運行該程序,在控制臺窗口工程的根目錄下輸入

    run world ( 其中"world"為調用例程中API的入口參數)

    經過一次web通訊,一兩秒后屏幕將顯示運行結果:


      hello,world!


    至此我們完成了一次Web服務的訪問過程。如果在程序執行過程中,我們用TCP Moniter之類的工具監視這一次訪問中的在網絡中流入流出的數據,可以看到客戶端發起調用的xml數據流如下:

    POST /sample/services/Sample1 HTTP/1.0
    Content-Type: text/xml; charset=utf-8
    Accept: application/soap+xml,
    application/dime, multipart/related, text/*
    User-Agent: Axis/1.1
    Host: 127.0.0.1
    Cache-Control: no-cache
    Pragma: no-cache
    SOAPAction: ""
    Content-Length: 430


      world




    服務器端返回的結果的XML輸出流如下:

    HTTP/1.1 200 OK
    Content-Type: text/xml; charset=utf-8
    Date: Wed, 31 Mar 2004 06:42:18 GMT
    Server: Apache Coyote/1.0
    Connection: close

     
      hello,world!
     

    結語

    以上詳細講解了Axis的序列化/反序列化器的開發過程,相信讀者已經從中學到了不少知識,并能夠應用于自己的項目開發中去。通過掌握這一技術,我們將更為深刻的理解Axis的內部結構和Web服務的工作機理,這些經驗是市面上那些泛泛的講解JAVA Web服務的參考書上所學不到的。后續的文章還將向您展示一些在Java Web服務深度開發中的高級技術,讓您真正駕馭Axis。

    參考資料

    附件中提供了本文中的全部源碼,它是一個完整的工程,可以參考學習。
    在apache的網站您可以獲得最新的Axis庫及其開發文檔:http://ws.apache.org/axis/ 本文中提到的Tomcat Web服務器可以在這里免費獲得:http://jakarta.apache.org/tomcat/index.html
    《IBM DEVELOPWORK》中一個關于web服務的教程《從JAVA類創建web服務》,是您快速掌握Web服務開發的很好教材,初學者可以進行學習。
    JDOM類庫可以去www.jdom.org下載,CSDN上有一篇不錯的文章可以讓您快速掌握JDOM的使用方法。http://www.csdn.net/develop/Read_Article.asp?Id=13806

    posted on 2007-08-30 12:42 菠蘿 閱讀(337) 評論(0)  編輯  收藏 所屬分類: AXIS

    主站蜘蛛池模板: 国产精品视频白浆免费视频| 99视频免费在线观看| 久草在视频免费福利| 亚洲视频一区二区三区| 久久黄色免费网站| 99人中文字幕亚洲区| 日本一卡精品视频免费| 亚洲福利一区二区精品秒拍| 91热成人精品国产免费| 国产 亚洲 中文在线 字幕| 午夜两性色视频免费网站| 美女视频黄.免费网址| 亚洲精品专区在线观看| 国产一级a毛一级a看免费视频| 亚洲国产精彩中文乱码AV| 日韩免费无码视频一区二区三区| 亚洲福利视频一区二区三区| 久久久久久国产精品免费免费 | 亚洲国产成人精品久久| 毛片免费视频观看| 国产99久久亚洲综合精品| 中文亚洲AV片在线观看不卡| 久久国产精品免费看| 国产精品亚洲片夜色在线| 国产高清免费在线| 在线观看人成视频免费无遮挡| 亚洲日本中文字幕区| 在线视频免费观看www动漫| 深夜A级毛片视频免费| 老汉色老汉首页a亚洲| 热99re久久免费视精品频软件| 国产免费AV片在线观看播放| 亚洲性天天干天天摸| 日韩高清在线免费观看| 国产精品无码免费专区午夜| 亚洲精品不卡视频| 免费国产a国产片高清| 中文字幕免费视频一| 国产亚洲综合视频| 亚洲成a人片在线网站| 亚洲男人在线无码视频|