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

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

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

    Terry.Li-彬

    虛其心,可解天下之問;專其心,可治天下之學;靜其心,可悟天下之理;恒其心,可成天下之業(yè)。

      BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
      143 隨筆 :: 344 文章 :: 130 評論 :: 0 Trackbacks

    SOAP協(xié)議規(guī)范

    1. 簡介

    SOAP以XML形式提供了一個簡單、輕量的用于在分散或分布環(huán)境中交換結構化和類型信息的機制。SOAP本身并沒有定義任何應用程序語義,如編程模型或特定語義的實現(xiàn);實際上它通過提供一個有標準組件的包模型和在模塊中編碼數(shù)據(jù)的機制,定義了一個簡單的表示應用程序語義的機制。這使SOAP能夠被用于從消息傳遞到RPC的各種系統(tǒng)。

    SOAP包括三個部分

    • SOAP封裝(見第4節(jié))結構定義了一個整體框架用來表示消息中包含什么內(nèi)容,誰來處理這些內(nèi)容以及這些內(nèi)容是可選的或是必需的。
    • SOAP編碼規(guī)則(見第5節(jié))定義了用以交換應用程序定義的數(shù)據(jù)類型的實例的一系列機制。
    • SOAP RPC表示(見第7節(jié))定義了一個用來表示遠程過程調(diào)用和應答的協(xié)定。

    雖然這三個部分都作為SOAP的一部分一起描述,但它們在功能上是相交的。特別的,封裝和編碼規(guī)則是在不同的名域中定義的,這種模塊性的定義方法增加了簡單性在SOAP封裝,SOAP編碼規(guī)則和SOAPRPC協(xié)定之外,這個規(guī)范還定義了兩個協(xié)議的綁定,描述了在有或沒有HTTP擴展框架[6]的情況下,SOAP消息如何包含在HTTP消息[5]中被傳送。

    1.1 設計目標

    SOAP的主要設計目標是簡單性和可擴展性,這意味著傳統(tǒng)的消息系統(tǒng)和分布對象系統(tǒng)的某些性質(zhì)不是SOAP規(guī)范的一部分。這些性質(zhì)包括:

    • 分布式碎片收集
    • 成批傳送消息
    • 對象引用(要求分布式碎片收集)
    • 激活機制(要求對象引用)

    1.2 符號約定

    這篇文章中的關鍵字 "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT","SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", 和"OPTIONAL"的解釋在RFC-2119 [2]中。 這篇文章中用到的名域前綴 "SOAP-ENV" 和"SOAP-ENC"分別與"http://schemas.xmlsoap.org/soap/envelope/" 和"http://schemas.xmlsoap.org/soap/encoding/"關聯(lián)。整篇文檔中,名域前綴“xsi”被假定為與URI"http://www.w3.org/1999/XMLSchema-instance“(在XMLSchema規(guī)范[11]定義)相連。類似的,名域前綴”xsd“被假定為與URI"http://www.w3.org/1999/XMLSchema"(在[10]中定義)相連。名域前綴”tns“用來表示任意名域。所有其它的名域前綴都只是例子。
    名域URI的基本形式”some-URI“表示某些依賴于應用程序或上下文的URI[4]。這個規(guī)范用擴展BNF(在RFC-2616[5] 描述)描述某些結構。

    1.3 SOAP消息舉例

    在這個例子中,GetLastTradePrice SOAP 請求被發(fā)往StockQuote服務。這個請求攜帶一個字符串參數(shù)和ticker符號,在SOAP應答中返回一個浮點數(shù)。XML名域用來區(qū)分SOAP標志符和應用程序特定的標志符。這個例子說明了在第6節(jié)中定義的HTTP綁定。如果SOAP中管理XML負載的規(guī)則完全獨立于HTTP是沒有意義的,因為事實上該負載是由HTTP攜帶的。在Appendix A中有更多的例子。

    例1 在HTTP請求中嵌入SOAP消息

    POST /StockQuote HTTP/1.1
    Host:
    www.stockquoteserver.com
    Content-Type: text/xml;
    charset="utf-8"
    Content-Length: nnnn
    SOAPAction:
    "Some-URI"
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/">
    <SOAP-ENV:Body>
    <m:GetLastTradePrice xmlns:m="Some-URI">
    <symbol>DIS</symbol>
    </m:GetLastTradePrice>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    下面是一條應答消息,包括HTTP消息,SOAP消息是其具體內(nèi)容 :

    例2 在HTTP應答中嵌入SOAP消息

    HTTP/1.1 200 OK
    Content-Type: text/xml;
    charset="utf-8"
    Content-Length:
    nnnn
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"/>
    <SOAP-ENV:Body>
    <m:GetLastTradePriceResponse xmlns:m="Some-URI">
    <Price>34.5</Price>
    </m:GetLastTradePriceResponse>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    2. SOAP消息交換模型

    SOAP消息從發(fā)送方到接收方是單向傳送,但正如上面顯示的,SOAP消息經(jīng)常以請求/應答的方式實現(xiàn)。SOAP實現(xiàn)可以通過開發(fā)特定網(wǎng)絡系統(tǒng)的特性來優(yōu)化。例如,HTTP綁定(見第6節(jié))使SOAP應答消息以HTTP應答的方式傳輸,并使用同一個連接返回請求。不管SOAP被綁定到哪個協(xié)議,SOAP消息采用所謂的”消息路徑“發(fā)送,這使在終節(jié)點之外的中間節(jié)點可以處理消息。一個接收SOAP消息的SOAP應用程序必須按順序執(zhí)行以下的動作來處理消息:識別應用程序想要的SOAP消息的所有部分 (見4.2.2節(jié))檢驗應用程序是否支持第一步中識別的消息中所有必需部分并處理它。如果不支持,則丟棄消息(見4.4節(jié))。在不影響處理結果的情況下,處理器可能忽略第一步中識別出的可選部分。如果這個SOAP應用程序不是這個消息的最終目的地,則在轉發(fā)消息之前刪除第一步中識別出來的所有部分。為了正確處理一條消息或者消息的一部分,SOAP處理器需要理解:所用的交換方式(單向,請求/應答,多路發(fā)送等等),這種方式下接收者的任務,RPC機制(如果有的話)的使用(如第7節(jié)中所述),數(shù)據(jù)的表現(xiàn)方法或編碼,還有其它必需的語義。盡管屬性(比如SOAP encodingstyle,見4.1.1節(jié))可以用于描述一個消息的某些方面,但這個規(guī)范并不 強制所有的接收方也必須有同樣的屬性并取同樣的屬性值。舉個例子,某一特定的應用可能知道一個元素表示一條遵循第7節(jié)約定的RPC請求,但是另外一些應用可能認為指向該元素的所有消息都用單向傳輸,而不是類似第7節(jié)的請求應答模式。
    (譯者注:交互雙方的SOAP消息并不一定要遵循同樣的格式設定,而只需要以一種雙方可理解的格式交換信息就可以了)

    3. 與XML的關系

    所有的SOAP消息都使用XML形式編碼(更多有關XML的信息請見[7])一個SOAP應用程序產(chǎn)生的消息中,所有由SOAP定義的元素和屬性中必須包括正確的名域。SOAP應用程序必須能夠處理它接收到的消息中的SOAP名域(見4.4節(jié)),并且它可以處理沒有SOAP名域的SOAP消息,就象它們有正確的名域一樣。SOAP定義了兩個名域(更多有關XML名域的信息請見[8])

    • SOAP封裝的名域標志符是"http://schemas.xmlsoap.org/soap/envelope/"
    • SOAP的編碼規(guī)則的名域標志符是"http://schemas.xmlsoap.org/soap/encoding/"

    SOAP消息中不能包含文檔類型聲明,也不能包括消息處理指令。[7] SOAP使用"ID"類型"id"屬性來指定一個元素的唯一的標志符,同時該屬性是局部的和無需校驗的。SOAP使用"uri-reference"類型的"href"屬性指定對這個值的引用,同時該屬性是局部的和無需校驗的。這樣就遵從了XML規(guī)范[7],XMLSchema規(guī)范[11]和XML連接語言規(guī)范[9]的風格。除了SOAP mustUnderstand 屬性(見4.2.3節(jié))和SOAPactor屬性(見4.2.2節(jié))之外,一般允許屬性和它們的值出現(xiàn)在XML文檔實例或Schema中(兩者效果相同)。也就是說,在DTD或Schema中聲明一個缺省值或固定值和在XML文檔實例中設置它的值在語義上相同。

    4. SOAP封裝

    SOAP消息是一個XML文檔,包括一個必需的SOAP封裝,一個可選的SOAP頭和一個必需的SOAP體。在這篇規(guī)范剩余部分中,提到SOAP消息時就是指這個XML文檔。這一節(jié)中定義的元素和屬性的名域標志符為:

    "http://schemas.xmlsoap.org/soap/envelope/" 。一個SOAP消息包括以下部分:1.在表示這個消息的XML文檔中,封裝是頂層元素。2.應用SOAP交換信息的各方是分散的且沒有預先協(xié)定,SOAP頭提供了向SOAP消息中添加關于這條SOAP消息的某些要素(feature)的機制。SOAP定義了少量的屬性用來表明這項要素(feature)是否可選以及由誰來處理。(見4.2節(jié))3.SOAP體是包含消息的最終接收者想要的信息的容器(見4.3節(jié))。SOAP為SOAP體定義了一個Fault元素用來報告錯誤信息。語法規(guī)則如下所示:

    封裝

    1. 元素名是 "Envelope"
    2. 在SOAP消息中必須出現(xiàn)。
    3. 可以包含名域聲明和附加屬性。如果包含附加屬性,這些屬性必須限定名域。類似的,"Envelope"可以包含附加子元素,這些也必須限定名域且跟在SOAP體元素之后。

    SOAP頭 (見4.2節(jié))

    1. 元素名是"Header"
    2. 在SOAP消息中可能出現(xiàn)。如果出現(xiàn)的話,必須是SOAP封裝元素的第一個直接子元素。
    3. SOAP頭可以包含多個條目,每個都是SOAP頭元素的直接子元素。所有SOAP頭的直接子元素都必須限定名域。

    SOAP體 (見4.3節(jié))

    1. 元素名是"Body"
    2. 在SOAP消息中必須出現(xiàn)且必須是SOAP封裝元素的直接子元素。它必須直接跟在SOAP頭元素(如果有)之后。否則它必須是SOAP封裝元素的第一個直接子元素。
    3. SOAP體可以包括多個條目,每個條目必須是SOAP體元素的直接子元素。SOAP體元素的直接子元素可以限定名域。SOAP定義了SOAPFault元素來表示錯誤信息。

    4.1.1 SOAP encodingStyle屬性

    EncodingStyle全局屬性用來表示SOAP消息的序列化規(guī)則。這個屬性可以在任何元素中出現(xiàn),作用范圍與名域聲明的作用范圍很相似,為這個元素的內(nèi)容和它的所有沒有重載此屬性的子元素。SOAP消息沒有定義缺省編碼。屬性值是一個或多個URI的順序列表,每個URI確定了一種或多種序列化規(guī)則,用來不同程度反序列化SOAP消息,舉例如下:

    "http://schemas.xmlsoap.org/soap/encoding/"
    "http://my.host/encoding/restricted http://my.host/encoding/"
    ""

    第5節(jié)中定義的序列化規(guī)則由URI"http://schemas.xmlsoap.org/soap/encoding/" 確定。使用這個特定序列化規(guī)則的消息應該用encodingStyle屬性說明這一點。另外,所有以"http://schemas.xmlsoap.org/soap/encoding/"開頭的URI中的序列化規(guī)則與第5節(jié)中定義的SOAP編碼規(guī)則相一致。一個零長度的URI("")明確顯示所含元素沒有任何編碼形式。這可以用來取消上一級元素的所有編碼聲明。

    4.1.2 封裝版本模型

    SOAP沒有定義常規(guī)的基于主版本號和輔版本號的版本形式。SOAP消息必須有一個封裝元素與名域"http://schemas.xmlsoap.org/soap/envelope/"關聯(lián)。如果SOAP應用程序接收到的SOAP消息中的SOAP封裝元素與其他的名域關聯(lián),則視為版本錯誤,應用程序必須丟棄這個消息。如果消息是通過HTTP之類的請求/應答協(xié)議收到的,應用程序必須回答一個SOAP VersionMismatch 錯誤信息(見4.4節(jié))。

    4.2 SOAP頭

    SOAP為相互通信的團體之間提供了一種很靈活的機制:在無須預先協(xié)定的情況下,以分散但標準的方式擴展消息。可以在SOAP頭中添加條目實現(xiàn)這種擴展,典型的例子有認證,事務管理,支付等等。頭元素編碼為SOAP封裝元素的第一個直接子元素。頭元素的所有直接子元素稱作條目。條目的編碼規(guī)則如下:

    一個條目有它的完整的元素名(包括名域URI和局部名)確定。SOAP頭的直接子元素必須有名域限制。
    SOAP encodingStyle屬性可以用來指示條目所用的編碼形式(見4.1.1節(jié))
    SOAP mustUnderstand屬性(見4.2.3節(jié))和SOAPactor屬性(見4.2.2節(jié))可以用來指示如何處理這個條目以及由誰來處理。(見4.2.1節(jié))

    4.2.1 使用頭屬性

    這一節(jié)中定義的SOAP頭屬性確定了SOAP消息的接收者應該怎樣按第2節(jié)中所述的方式處理消息。產(chǎn)生SOAP消息的SOAP應用程序,應該僅僅在SOAP頭元素的直接子元素中使用這些SOAP頭屬性。SOAP消息的接收者必須忽略所有不在SOAP頭元素的直接子元素中SOAP頭屬性。下面的例子是一個SOAP頭,包括一個元素標志符"Transaction","mustUnderstand"取值為"1"和數(shù)值5。這應該以如下方式編碼:

    <SOAP-ENV:Header>
    <t:Transaction
    xmlns:t="some-URI" SOAP-ENV:mustUnderstand="1">
    5
    </t:Transaction>
    </SOAP-ENV:Header>

    4.2.2 SOAP actor屬性

    一個SOAP消息從始節(jié)點到終節(jié)點的過程中,可能沿著消息路徑經(jīng)過一系列SOAP中間節(jié)點。一個SOAP中間節(jié)點是一個可以接收轉發(fā)SOAP消息的應用程序。中間節(jié)點和終節(jié)點由URI區(qū)分。可能SOAP消息的終節(jié)點并不需要所有部分,而在消息路徑上的一個和幾個中間節(jié)點可能需要這些內(nèi)容。頭元素的接收者扮演的角色類似于一個過濾器,防止這些只發(fā)給本接受者的消息部分擴散到其它節(jié)點。即一個頭元素的接收者必須不轉發(fā)這些頭元素到SOAP消息路徑上的下一個應用程序。同樣的,接收者可能插入一個相似的頭元素。SOAP actor全局屬性可以用于指示頭元素的接收者。SOAP actor屬性的值是一個URI。

    URI "http://schemas.xmlsoap.org/soap/actor/next"指出了第一個處理這個消息的SOAP應用程序需要這個頭元素。這類似于HTTP頭中用Connection域表示hop-by-hop范圍模型。省略SOAP actor屬性表示接收者是SOAP消息的終節(jié)點。如果這個屬性要生效,它必須出現(xiàn)在SOAP消息實例中。(見第3節(jié)和4.2.1節(jié))

    4.2.3 SOAP mustUnderstand屬性

    SOAP mustUnderstand全局屬性用來指示接受者在處理消息時這個條目是否必須處理。條目的接收者由SOAP actor屬性定義(見4.2.2節(jié))。MustUnderstand屬性的值是"1" 或 "0"。缺少SOAP mustUnderstand屬性在語義上等同于它的值為"0"。如果一個頭元素的SOAP mustUnderstand屬性的值是"1",那么條目的接受者必須或者遵守語義(如以元素的全名傳送)并按照語義正確的處理,或者放棄處理消息(見4.4節(jié))。SOAP mustUnderstand 屬性考慮了消息演變的準確性(robust evolution)。必須假定包含SOAP mustUnderstand屬性且值為"1"的元素以某種方式修改了它們的父元素或同層元素的語義。以這種方式連接元素確保了語義上的變化不會被那些不能完全理解它的接收者忽略。如果這個屬性要生效,它必須出現(xiàn)在SOAP消息實例中。(見第3節(jié)和4.2.1節(jié))

    4.3 SOAP體

    SOAP體元素提供了一個簡單的機制,使消息的最終接收者能交換必要的信息。使用體元素的典型情況包括配置RPC請求和錯誤報告。體元素編碼為SOAP封裝元素的直接子元素。如果已經(jīng)有一個頭元素,那么體元素必須緊跟在頭元素之后,否則它必須是SOAP封裝元素的第一個直接子元素。體元素的所有直接子元素稱作體條目,每個體條目在SOAP體元素中編碼為一個獨立的元素。條目的編碼規(guī)則如下:

    • 一個條目由它的元素全名(包括名域URI和局部名)確定。SOAP體元素的直接子元素可能是名域限制的。
    • SOAP encodingStyle屬性可能用來指示條目(見4.1.1節(jié))的編碼方式。
    • SOAP定義了一個Fault條目用來報告錯誤信息。(見4.4節(jié))

    4.3.1 SOAP頭和體的關系

    雖然頭和體定義為獨立的元素,它們實際上是有關系的。體條目和頭條目的關系如下:體條目在語義上等同于actor屬性為缺省值且mustUnderstand屬性值為"1"的頭條目。不使用actor屬性則表示缺省的actor。(見4.2.2節(jié))

    4.4 SOAP錯誤

    SOAP錯誤元素用于在SOAP消息中攜帶錯誤和(或)狀態(tài)信息。如果有SOAP錯誤元素,它必須以以體條目的方式出現(xiàn),并且在一個體元素中最多出現(xiàn)一次。SOAP錯誤元素定義了以下四個子元素:

    • faultcode
      faultcode元素給軟件提供了一個識別此錯誤的算法機制。SOAP錯誤元素必須有faultcode子元素,并且它的值必須是一個合法的名(在[8]節(jié)定義)。SOAP定義一些SOAP faultcode描述基本的SOAP錯誤(見4.4.1節(jié))。
    • faultstring
      faultstring元素提供了一個錯誤解釋,而不是為了軟件處理。faultstring元素類似于HTTP中定義(見[5],第6.1節(jié))的'Reason-Phrase'。SOAP錯誤元素必須有faultstring子元素,并且它應該提供一些錯誤本質(zhì)的解釋信息。
    • faultactor
      faultactor元素提供了在消息路徑上是誰導致了錯誤發(fā)生的信息(見第2節(jié))。它類似于SOAP actor屬性(見4.2.2節(jié)),只是SOAP actor指的是頭條目的目的地,faultactor指的是錯誤的來源。faultactor屬性的值是用來區(qū)分錯誤來源的URI。不是SOAP消息的最終目的地的應用程序必須在SOAP Fault元素中包含faultactor元素。消息的最終目的地可以使用faultactor元素明確的指示是它產(chǎn)生了這個錯誤(參見下面的detail元素)
    • detail
      detail元素用來攜帶與Body元素有關的應用程序所要的錯誤信息。如果Body元素的內(nèi)容不能被成功的處理,則必須包含detail子元素。它不能用來攜帶屬于頭條目的錯誤信息。頭條目的詳細出錯信息必須由頭條目攜帶。Fault元素中沒有detail元素表示這個錯誤與Body元素的處理無關。在有錯誤的時候,這可以用來區(qū)分Body元素有沒有被正確的處理。detail元素的所有直接子元素稱作detail條目,并且每個detail條目在detail元素中編碼為獨立的元素。detail條目的編碼規(guī)則如下(參見例10): 一個detail條目由它的元素全名(包括名域URI和局部名)確定。SOAP體元素的直接子元素可能是名域限制的。SOAP encodingStyle屬性可能用來指示detail條目(見4.1.1節(jié))的編碼方式。也可以有其它的Fault子元素,只要它們是名域限制的。

    4.4.1 SOAP 錯誤代碼

    在描述這個規(guī)范中定義的錯誤時,這一節(jié)中定義的Faultcode值必須用在faultcode元素中。這些faultcode值得名域標志符為"http://schemas.xmlsoap.org/soap/envelope/"。定義這個規(guī)范之外的方法時推薦(不要求)使用這個名域。缺省的SOAP faultcode值以可擴展的方式定義,允許定義新的SOAP faultcode值,并與現(xiàn)有的faultcode值向后兼容。使用的機制類似于HTTP中定義的1xx, 2xx,3xx等基本的狀態(tài)類(見[5]第10節(jié)),不過,它們定義為XML合法名(見 [8] 第3節(jié) ),而不是整數(shù)。 字符"."(點)作為faultcode的分隔符,點左邊的錯誤代碼比右邊的錯誤代碼更為普通。如:

    Client.Authentication

    這篇文檔中定義的faultcode值是:

    名稱 含義
    VersionMismatch 處理方發(fā)現(xiàn)SOAP封裝元素有不合法的名域(見4.1.2節(jié))
    MustUnderstand 處理方不理解或者不服從一個包含值為"1"的
    mustUnderstand 屬性的 SOAP頭元素的直接子元素。(見4.2.3節(jié))

    Client

    Client錯誤類表示消息的格式錯誤或者不包含適當?shù)恼_信息。例如,消息可能缺少正確的認證和支付信息。一般地,它表示消息不能不作修改就重發(fā)。參見4.4節(jié)

    SOAP Fault detail子元素的描述。

    Server

    Server錯誤類表示由于消息的處理過程而不是消息的內(nèi)容本身使得消息消息不能正確的處理。例如,處理消息時可能要與其它處理器通信,但它沒有響應。這個消息可能在遲一點的時間處理成功。 SOAP Fault子元素的詳細信息參見4.4節(jié)

    5. SOAP編碼

    SOAP編碼格式基于一個簡單的類型系統(tǒng),概括了程序語言,數(shù)據(jù)庫和半結構化數(shù)據(jù)等類型系統(tǒng)的共同特性。一個類型或者是一個簡單的(標量的)類型,或者是由幾個部分組合而成的復合類型,其中每個部分都有自己的類型。以下將詳細描述這些類型。這一節(jié)定義了類型化對象的序列化規(guī)則。它分兩個層次。首先,給定一個與類型系統(tǒng)的符號系統(tǒng)一致的Schema(譯者注:這里的schema不是符合XML語法的schema,而僅僅表示廣義的用于表示消息結構的定義方式),就構造了XML語法的Schema。然后,給定一個類型系統(tǒng)的Schema和與這個Schema一致的特定的值,就構造了一個XML文檔實例。反之,給定一個依照這些規(guī)則產(chǎn)生的XML文檔實例和初始的Schema,就可以構造初始值的一個副本。這一節(jié)中定義的元素和屬性的名域標志符為"http://schemas.xmlsoap.org/soap/encoding/"。下面的例子都假定在上一層的元素中聲明了名域。
    鼓勵使用這一節(jié)中描述的數(shù)據(jù)模型和編碼方式,但也可以在SOAP中使用其他的數(shù)據(jù)模型和編碼方式。(見4.1.1節(jié))

    5.1 XML中的編碼類型規(guī)則

    XML允許非常靈活的數(shù)據(jù)編碼方式。SOAP定義了一個較小的規(guī)則集合。這一節(jié)在總的層次上定義了這些編碼規(guī)則,下一節(jié)將描述特定類型的編碼規(guī)則的細節(jié)。這一節(jié)定義的編碼規(guī)則可以與第7節(jié)中所述的RPC調(diào)用和應答映射結合使用。下面的術語用來描述編碼規(guī)則:

    • 一個"value"是一個字符串,類型(數(shù)字,日期,枚舉等等)的名或是幾個簡單值的組合。所有的值都有特定的類型。
    • 一個"simple value"沒有名部分, 如特定的字符串,整數(shù),枚舉值等等。
    • 一個"compound value"是相關的值的結合,如定單,股票報表,街道地址等等。在"compound value"中,每個相關的值都潛在的以名,序數(shù)或這兩者來區(qū)分。這叫作"a ccessor"。復合值的例子有定單和股票報表等等。數(shù)組也是復合值。在復合值中,多個accessor有相同的名是允許的,例如RDF就是這樣做的。
    • 一個"array"是一個復合值,成員值按照在數(shù)組中的位置相互區(qū)分。
    • 一個"struct"也是一個復合值,成員值之間的唯一區(qū)別是accessor名,accessor名互不相同。
    • 一個"simple type"是簡單值的類,如叫做"string" "integer"的類,還有枚舉類等等。
    • 一個"compound type"是復合值的類。復合類型的例子有定單類,它們有相同的accessor名(shipTo, totalCost等),但可能會有不同的值(可能以后被設置為確定的值)。

    在復合類型中,如果類型內(nèi)的accessor名互不相同,但是可能與其他類型中的accessor名相同,即,accessor名加上類型名形成一個唯一的標志符,這個名叫作"局部范圍名"。如果名是直接或間接的基于URI的一部分,那么不管它出現(xiàn)在什么類型中,這個名本身就可以唯一標志這個accessor,這樣的名叫作"全局范圍名"。給定了schema中相關的值的序列化信息,就可能確定某些值只與某個accessor的一個實例有關。其它情況下則無法確定。當且僅當一個accessor引用一個值,這個值才能被視為"single-reference",如果有不止一個accessor引用它,那么就將它視為"multi-reference"。注意,可能一個確定的值在一個schema中是"single-reference",而在另一個schema中是"multi-reference"。在語句構成上,一個元素可能是"independent" 或 "embedded"。一個獨立的元素指出現(xiàn)在序列化最頂層的任何元素。所有其它元素都是嵌入元素。雖然用xsi:type屬性可以使值的結構和類型變?yōu)樽悦枋龅模切蛄谢?guī)則允許值的類型僅僅參照schema而定。這樣的schema可能使用"XML Schema Part 1: Structures" [10]和"XML Schema Part 2: Datatypes" [11]中描述的符號系統(tǒng),也可能使用其它符號系統(tǒng)。注意,雖然序列化規(guī)則可以用于除了數(shù)組和結構之外的復合類型,但是許多schema僅僅包含數(shù)組和結構類型。序列化規(guī)則如下:

    所有的值以元素內(nèi)容的形式表示。一個multi-reference值必須表示為一個獨立元素的內(nèi)容,而一個single-reference值最好不要這樣表示(也可以這樣表示)。對于每個具有值的元素,值的類型時必須用下述三種方式之一描述:

    • 所屬元素實例有xsi:type屬性
    • 所屬元素是一個有SOAP-ENC:arrayType 屬性(該屬性可能是缺省的)的元素的子元素,或者
    • 所屬元素的名具有特定的類型,類型可以由schema確定。

    一個簡單值表示為字符數(shù)據(jù),即沒有任何子元素。每個簡單值必須具有一個類型,這個類型或者是XML Schemas Specification, part 2 [11]有的類型,或者具有源類型(參見5.2節(jié))。一個復合值編碼成一個元素的序列,每個accessor用一個嵌入元素表示,該元素的元素名和accessor的名一致。如果accessor的名是局部于其所屬的類型的,則該元素的元素名不是合格的,否則對應的元素名是合格的。(參見5.4節(jié))
    一個multi-reference的簡單值或復合值編碼成一個獨立的元素,這個元素包含一個局部的無需校驗的屬性,屬性名為"id",類型為"ID"(依照XML Specification [7])。值的每個accessor對應一個空元素,該元素有一個局部的,無需校驗的屬性,屬性名為"href",類型為" uri-reference "(依照XML Schema Specification [11]),"href"屬性的值引用了相對應的獨立元素的URI標志符。字符串和字符數(shù)組表示為multi-reference的簡單類型,但是特殊的規(guī)則使它們在普通的情況下能被更有效的表示(參見5.2.1節(jié)和5.2.3節(jié))。字符串和字符數(shù)組值的accessor可能有一個名字為"id",類型為"ID"(依照XML Specification [7])的屬性。如果這樣,所有這個值的所有其它accessor編碼成一個空元素,這個元素有一個局部的,無需校驗的屬性,屬性名為"href",類型為" uri-reference "(依照XML Schema Specification [11]),"href"屬性的值引用了包含這個值的元素的URI標志符。編碼時允許一個值有多個引用,就像多個不同的值有多個引用一樣,但這僅在從上下文可以知道這個XML文檔實例的含義沒有改變時才可使用。數(shù)組是復合值(參見5.4.2節(jié))。SOAP數(shù)組定義為具有類型"SOAP-ENC:Array"或從它衍生的類型.

    SOAP數(shù)組可以時一維或多維,它們的成員以序數(shù)位置相互區(qū)分。一個數(shù)組值表示為反映這個數(shù)組的一系列元素,數(shù)組成員按升序出現(xiàn)。對多維數(shù)組來說,右邊的這一維變化最快。每個成員元素命名為一個獨立元素。(見規(guī)則2)SOAP數(shù)組可以是single-reference 或multi-reference值,因此可以表示為嵌入元素或獨立元素的內(nèi)容。SOAP數(shù)組必須包含一個"SOAP-ENC:arrayType"屬性,它的值指定了包含元素的類型和數(shù)組的維數(shù)。"SOAP-ENC:arrayType"屬性的值定義如下:

    arrayTypeValue = atype asize
    atype = QName *( rank )
    rank = "[" *( "," ) "]"
    asize = "[" #length "]"
    length = 1*DIGIT

    • "atype"結構是被包含元素的類型名,它表示為QName并且作為類型限制在XML元素聲明的
    • "type"屬性中出現(xiàn)(這意味著被包含元素的所有值都要與該類型一致,即在SOAP-ENC:a rrayType中引用的類型必須是每個數(shù)組成員的類型或超類型)。在arrays of arrays or "jagged arrays"的情況下,類型組件編碼為"innermost"類型且在從第一層開始的嵌套數(shù)組的每一層中,類型名后都跟隨一個rank結構。多維數(shù)組編碼時從第一維起,每一維之間用逗號隔開。
    • "asize"結構包含一個以逗號分隔的列表,數(shù)值0,1或其它整數(shù)表示數(shù)組每一維的長度。整數(shù)0表示沒有指定詳細的大小,但是可能在檢查數(shù)組實際成員的大小后確定。例如,一個5個成員的整型數(shù)組的arrayTypeValue值為"int[][5]",它的atype值是int[]",asize值是"[5]"。同樣,一個3個成員的兩維整型數(shù)組的arrayTypeValue值為"int[,][3]",它的atype值是int[,]",asize值是"[3]"。

    一個SOAP數(shù)組成員可能包含一個"SOAP-ENC:offset"屬性表示這一項在整個數(shù)組中的位置偏移值。這被用來指示一個部分儲值數(shù)組(見5.4.2.1節(jié))的位置偏移值。同樣,一個數(shù)組成員可能包含一個"SOAP-ENC:position"屬性表示這一項在整個數(shù)組中的位置,這被用來描述稀疏數(shù)組(見5.4.2.2節(jié))的成員。"SOAP-ENC:offset" 和"SOAP-ENC:position"屬性值的定義如下:

    arrayPoint = "[" #length "]"
    偏移值和位置從0開始
    NULL值或缺省值可能通過省略accssor元素來表示。NULL值也可能通過一個包含值為'1'的xsi:null屬性的accssor元素來表示,其它的依賴于應用程序的屬性和值也可能用來表示NULL值。注意,規(guī)則2允許獨立的元素和數(shù)組成員名不同于值類型的元素。

    5.2 簡單類型

    SOAP采用了"XML Schema Part 2: Datatypes"規(guī)范[11]"Built-in datatypes"節(jié)中的所有類型作為簡單類型,包括值和取值范圍。例如:

    類型 舉例
    int 58502
    float 314159265358979E+1
    negativeInteger -32768
    string Louis "Satchmo" Armstrong

    在XML Schema規(guī)范中聲明的數(shù)據(jù)類型可以直接用在元素schema中,也可以使用從這些類型衍生的新類型。一個schema和對應的具有這些類型的元素的數(shù)據(jù)實例的例子如下所示:

    <element name="age" type="int"/>
    <element name="height" type="float"/>
    <element name="displacement" type="negativeInteger"/>
    <element name="color">
    <simpleType base="xsd:string">
    <enumeration value="Green"/>
    <enumeration value="Blue"/>
    </simpleType>
    </element>
    <age>45</age>
    <height>5.9</height>
    <displacement>-450</displacement>
    <color>Blue</color>

    所有簡單值必須編碼為元素的內(nèi)容,它的類型或者在"XML Schema Part 2: Datatypes"規(guī)范[11]中定義過,或者是基于一個用XML Schema規(guī)范提供的機制能推衍生出的類型。如果一個簡單值編碼為獨立元素或異質(zhì)數(shù)組成員,那么有一個對應于數(shù)據(jù)類型的元素聲明將會很方便。因為"XML Schema Part 2: Datatypes"規(guī)范[11]包括了類型定義,但是不包括對應的元素聲明,SOAP-ENC schema和名域為每個簡單數(shù)據(jù)類型聲明了一個元素,如<SOAP-ENC:int id="int1">45</SOAP-ENC:int>

    5.2.1 字符串

    字符串數(shù)據(jù)類型的定義在"XML Schema Part 2: Datatypes"規(guī)范[11]中。注意,這不同于許多數(shù)據(jù)庫和程序語言中的"string"類型,特別的,字符串數(shù)據(jù)類型可能禁止某些在那些語言中允許的字符。(這些值必須用xsd:string之外的數(shù)據(jù)類型表示)一個字符串可能編碼為一個single-reference 或 multi-reference值。包含字符串值的元素可能有一個"id"屬性。附加的accessor元素可能有對應的"href"屬性。
    例如,同一字符串的兩個accessor可能以如下形式出現(xiàn):

    <greeting id="String-0">Hello</greeting>
    <salutation href="#String-0"/>

    但是,如果兩個accessor參考同一字符串實例(或字符串的子類型),這不是一個實質(zhì)問題,它們可以編碼為兩個single-reference值,如下所示:

    <greeting>Hello</greeting>
    <salutation>Hello</salutation>

    這個例子的schema片斷如下所示:

    <element name="greeting" type="SOAP-ENC:string"/>
    <element name="salutation" type="SOAP-ENC:string"/>

    在這個例子中,SOAP-ENC:string類型用作元素的類型,這是聲明數(shù)據(jù)類型是"xsd:string"且允許"id" 和"href"屬性的元素的簡便方法。精確定義參見SOAP編碼schema。Schemas可以使用這些源自SOAP編碼schema的聲明,但也可以不這樣做。

    5.2.2 Enumerations

    "XML Schema Part 2: Datatypes"規(guī)范 [11] 定義了"enumeration."機制。SOAP數(shù)據(jù)模型直接采用了這種機制。但是,由于程序語言和其它語言在定義枚舉時通常有些不同,所以我們在這里詳細闡述了它的概念并描述了一個列表成員的可能取的值是如何編碼的。"Enumeration"作為一個概念表示不同的名字的集合。一個特定的枚舉就是對應于特定的基類型的不同的值的列表。例如,顏色集合("Green", "Blue", "Brown")可以定義為基于字符串類型的枚舉,("1", "3", "5")可能是一個基于整型數(shù)的枚舉,等等。"XML Schema Part 2: Datatypes" [11]支持除了布爾型以外所有簡單類型的枚舉。"XML Schema Part 1: Structures"規(guī)范[10]的語言可以用來定義枚舉類型。如果schema由另一個沒有特定基類型適用的符號系統(tǒng)生成,就使用"string"。在下面schema的例子中,"EyeColor"定義為字符串,可能的值是"Green", "Blue", 或"Brown"的枚舉,數(shù)據(jù)實例按照schema顯示如下。

    <element name="EyeColor" type="tns:EyeColor"/>
    <simpleType name="EyeColor" base="xsd:string">
    <enumeration value="Green"/>
    <enumeration value="Blue"/>
    <enumeration value="Brown"/>
    </simpleType>
    <Person>
    <Name>Henry Ford</Name>
    <Age>32</Age>
    <EyeColor>Brown</EyeColor>
    </Person>

    5.2.3 字符數(shù)組

    一個字符數(shù)組可能編碼為single-reference 或multi-reference值。字符數(shù)組的編碼規(guī)則與字符串的編碼規(guī)則類似。特別的,包含字符數(shù)組的元素值可能由一個"id"屬性,附加的accssor元素可能有相應的"href"屬性。推薦使用定義在XML Schemas [10][11]中的'base64'編碼(使用在2045 [13]中定義的base64編碼算法)表示模糊字符數(shù)組。不過,由于行長度(line length)的限制,通常在MIME中應用base64編碼,SOAP中一般不應用base64編碼。但是提供了"SOAP-ENC:base64"子類型使之能用于SOAP。

    <picture xsi:type="SOAP-ENC:base64">
    aG93IG5vDyBicm73biBjb3cNCg==
    </picture>

    5.3 多態(tài)accessor

    許多語言允許能夠多態(tài)訪問多種類型值的accessor,每種類型在運行時可用。一個多態(tài)accessor實例必須包含一個"xsi:type"屬性描述實際值的類型。例如,一個名為"cost"類型值為"xsd:float"的多態(tài)accessor編碼如下:

    <cost xsi:type="xsd:float">29.95</cost>與之對比,類型值不變的accessor編碼如下:

    <cost>29.95</cost>

    5.4 Compound types復合類型

    SOAP定義了與下列常在程序語言中出現(xiàn)的結構性模式對應的類型:

    • 結構:一個"struct"是一個復合值,它的成員值的唯一區(qū)別是accessor名稱,任意兩個accessor名稱都不相同。
    • 數(shù)組:一個"array"是一個復合值,它的成員值的唯一區(qū)別是序數(shù)位置。

    SOAP也允許結構和數(shù)組之外的其它數(shù)據(jù)的序列化,例如Directed-Labeled-Graph Data Model之類的數(shù)據(jù)中,單個節(jié)點有許多不同的accssor,有些不止出現(xiàn)一次。SOAP序列化規(guī)則不要求底層的數(shù)據(jù)模型在accssor之間區(qū)分次序,但如果有這樣的次序的話,這些accssor必須按照這個順序編碼。

    5.4.1 復合值,結構和值引用

    復合值的成員編碼為accessor元素。當accessor由名區(qū)分時(如結構),accessor名即作為元素名。名局部于類型的accessor有不受限的名,其它的accessor則有受限的名。下面的例子是類型為"Book"的結構:

    <e:Book>
    <author>Henry Ford</author>
    <preface>Prefatory text</preface>
    <intro>This is a book.</intro>
    </e:Book>

    以下是描述上面結構的schema片斷:

    <element name="Book">
    <complexType>
    <element name="author" type="xsd:string"/>
    <element name="preface" type="xsd:string"/>
    <element name="intro" type="xsd:string"/>
    </complexType>
    </e:Book>

    以下是一個同時具有簡單和復雜成員類型的例子。它顯示兩層引用。注意"Author"accssor元素的"href"屬性是對相應具有"id"屬性的值的引用。"Address"與之類似。

    <e:Book>
    <title>My Life and Work</title>
    <author href="#Person-1"/>
    </e:Book>
    <e:Person id="Person-1">
    <name>Henry Ford</name>
    <address href="#Address-2"/>
    </e:Person>
    <e:Address id="Address-2">
    <email>mailto:henryford@hotmail.com</email>
    <web>http://www.henryford.com</web>
    </e:Address>

    當"Person"的值和"Address"的值是multi-reference時,上面的形式是正確的。如果它
    們是single-reference,就必須用嵌入的形式,如下所示:

    <e:Book>
    <title>My Life and Work</title>
    <author>
    <name>Henry Ford</name>
    <address>
    <email>mailto:henryford@hotmail.com</email>
    <web>http://www.henryford.com</web>
    </address>
    </author>
    </e:Book>

    如果添加一個限制,任意兩個人都不會有相同的地址,并且地址可以是街道或Email地址,一本書可以有兩個作者,編碼如下:

    <e:Book>
    <title>My Life and Work</title>
    <firstauthor href="#Person-1"/>
    <secondauthor href="#Person-2"/>
    </e:Book>
    <e:Person id="Person-1">
    <name>Henry Ford</name>
    <address xsi:type="m:Electronic-address">
    <email>mailto:henryford@hotmail.com</email>
    <web>http://www.henryford.com</web>
    </address>
    </e:Person>
    <e:Person id="Person-2">
    <name>Samuel Crowther</name>
    <address xsi:type="n:Street-address">
    <street>Martin Luther King Rd</street>
    <city>Raleigh</city>
    <state>North Carolina</state>
    </address>
    </e:Person>

    序列化可以包含對不在同一個資源的值的引用:

    <e:Book>
    <title>Paradise Lost</title>
    <firstauthor />
    </e:Book>

    以下是描述上面結構的schema片斷:

    <element name="Book" type="tns:Book"/>
    <complexType name="Book">
    <!-- Either the following group must occur or else the
    href attribute must appear, but not both. -->
    <sequence minOccurs="0" maxOccurs="1">
    <element name="title" type="xsd:string"/>
    <element name="firstauthor" type="tns:Person"/>
    <element name="secondauthor" type="tns:Person"/>
    </sequence>
    <attribute name="href" type="uriReference"/>
    <attribute name="id" type="ID"/>
    <anyAttribute namespace="##other"/>
    </complexType>
    <element name="Person" base="tns:Person"/>
    <complexType name="Person">
    <!-- Either the following group must occur or else the
    href attribute must appear, but not both. -->
    <sequence minOccurs="0" maxOccurs="1">
    <element name="name" type="xsd:string"/>
    <element name="address" type="tns:Address"/>
    </sequence>
    <attribute name="href" type="uriReference"/>
    <attribute name="id" type="ID"/>
    <anyAttribute namespace="##other"/>
    </complexType>
    <element name="Address" base="tns:Address"/>
    <complexType name="Address">
    <!-- Either the following group must occur or else the
    href attribute must appear, but not both. -->
    <sequence minOccurs="0" maxOccurs="1">
    <element name="street" type="xsd:string"/>
    <element name="city" type="xsd:string"/>
    <element name="state" type="xsd:string"/>
    </sequence>
    <attribute name="href" type="uriReference"/>
    <attribute name="id" type="ID"/>
    <anyAttribute namespace="##other"/>
    </complexType>

    5.4.2 數(shù)組

    SOAP數(shù)組定義為具有"SOAP-ENC:Array"類型或一個從"SOAP-ENC:Array"衍生的類型(參見規(guī)則8)。數(shù)組表示為元素值,對元素的名沒有特別的約束(正如元素值并不約束它們所屬的元素)。數(shù)組可以包含任意類型的元素,包括嵌套數(shù)組。可以創(chuàng)建新的類型(受SOAP-ENC:Array
    類型限制)來表示數(shù)組,如整數(shù)數(shù)組或某些用戶定義的枚舉。數(shù)組值表示為組成這個數(shù)組的項的元素的規(guī)則序列。在數(shù)組值中,元素名對于區(qū)分accesor并不重要。元素可以有任意的名。實際上,元素常常用它們在schema中暗示或確定的數(shù)組類型來命名元素。并且一般情況下對于復合值來說,如果數(shù)組中數(shù)組項的值是single-reference值,則這個數(shù)組項包含它的值,否則,該數(shù)組項通過"href"屬性引用這個值。下面的例子是一個整型數(shù)組的schema片斷:

    <element name="myFavoriteNumbers"
    type="SOAP-ENC:Array"/>
    <myFavoriteNumbers
    SOAP-ENC:arrayType="xsd:int[2]">
    <number>3</number>
    <number>4</number>
    </myFavoriteNumbers>

    在這個例子中,數(shù)組"myFavoriteNumbers"包括幾個成員,每個成員是一個類型為SOAP-ENC:int的值。注意SOAP-ENC:Array允許不受限制的元素名,它們不傳達任何類型信息,所以在使用時,或者它們有xsi:type屬性,或者它們所屬的元素有SOAP-ENC:arrayType屬性。自然,由SOAP-ENC:Array衍生的類型可以聲明局部元素,但這種情況下要包括類型信息。上面已經(jīng)提到,SOAP-ENC schema包含了元素的聲明,元素名與"XML Schema Part 2: Datatypes"規(guī)范[11]中的簡單類型一致。其中包括了對"Array"的聲明。于是,我們可以這樣寫:

    <SOAP-ENC:Array SOAP-ENC:arrayType="xsd:int[2]">
    <SOAP-ENC:int>3</SOAP-ENC:int>
    <SOAP-ENC:int>4</SOAP-ENC:int>
    </SOAP-ENC:Array>

    數(shù)組可以包含特定arrayType的任意子類型的實例。即,數(shù)組成員可以是arryType屬性值指定的類型的任意子類型,這個類型對于arrayType屬性中指定的類型來說是可替換的(根據(jù)schema中的替換規(guī)則)。例如,一個整型數(shù)組可以包含從整型衍生的任意類型(如"int"或任意用戶定義的從整型衍生的類型)。同樣,一個"address"數(shù)組可能包含一個address的受限類型或擴展類型如"internationalAddress"。因為提供的SOAP-ENC:Array類型允許任意類型的成員,所以可以包含任意類型的混合除非使用arrayType屬性加以特別的限制。在實例中,可以使用xsi:type指定成員元素的類型,或通過schema中成員元素的聲明來指定。下面是兩個例子。

    <SOAP-ENC:Array SOAP-ENC:arrayType="SOAP-ENC:ur-type[4]">
    <thing xsi:type="xsd:int">12345</thing>
    <thing xsi:type="xsd:decimal">6.789</thing>
    <thing xsi:type="xsd:string">
    Of Mans First Disobedience, and the Fruit
    Of that Forbidden Tree, whose mortal tast
    Brought Death into the World, and all our woe,
    </thing>
    <thing xsi:type="xsd:uriReference"> http://www.dartmouth.edu/~milton/reading_room/ </thing>
    </SOAP-ENC:Array>
    <SOAP-ENC:Array SOAP-ENC:arrayType="SOAP-ENC:ur-type[4]">
    <SOAP-ENC:int>12345</SOAP-ENC:int>
    <SOAP-ENC:decimal>6.789</SOAP-ENC:decimal>
    <xsd:string>
    Of Mans First Disobedience, and the Fruit
    Of that Forbidden Tree, whose mortal tast
    Brought Death into the World, and all our woe,
    </xsd:string>
    <SOAP-ENC:uriReference> http://www.dartmouth.edu/~milton/reading_room/ </SOAP-ENC:uriReference >
    </SOAP-ENC:Array>

    數(shù)組值可以是結構或其它復合值。例如"xyz:Order"結構數(shù)組:

    <SOAP-ENC:Array SOAP-ENC:arrayType="xyz:Order[2]">
    <Order>
    <Product>Apple</Product>
    <Price>1.56</Price>
    </Order>
    <Order>
    <Product>Peach</Product>
    <Price>1.48</Price>
    </Order>
    </SOAP-ENC:Array>

    數(shù)組成員值也可以是數(shù)組。下例是兩個字符串數(shù)組組成的數(shù)組:

    <SOAP-ENC:Array SOAP-ENC:arrayType="xsd:string[][2]">
    <item href="#array-1"/>
    <item href="#array-2"/>
    </SOAP-ENC:Array>
    <SOAP-ENC:Array id="array-1" SOAP-ENC:arrayType="xsd:string[2]">
    <item>r1c1</item>
    <item>r1c2</item>
    <item>r1c3</item>
    </SOAP-ENC:Array>
    <SOAP-ENC:Array id="array-2" SOAP-ENC:arrayType="xsd:string[2]">
    <item>r2c1</item>
    <item>r2c2</item>
    </SOAP-ENC:Array>

    包含數(shù)組的元素無需命名為"SOAP-ENC:Array"。它可以有任意的名,只要元素的類型是SOAP-ENC:Array或由之衍生的類型。例如,下面是一個schema片斷和與之一致的數(shù)組實例。

    <simpleType name="phoneNumber" base="string"/>
    <element name="ArrayOfPhoneNumbers">
    <complexType base="SOAP-ENC:Array">
    <element name="phoneNumber" type="tns:phoneNumber" maxOccurs="unbounded" />
    </complexType>
    <anyAttribute/>
    </element>
    <xyz:ArrayOfPhoneNumbers SOAP-ENC:arrayType="xyz:phoneNumber[2]">
    <phoneNumber>206-555-1212</phoneNumber>
    <phoneNumber>1-888-123-4567</phoneNumber>
    </xyz:ArrayOfPhoneNumbers>

    數(shù)組可能是多維的。在這種情況下,在arrayType屬性的asize部分將不止有一個值:

    <SOAP-ENC:Array SOAP-ENC:arrayType="xsd:string[2,3]">
    <item>r1c1</item>
    <item>r1c2</item>
    <item>r1c3</item>
    <item>r2c1</item>
    <item>r2c2</item>
    <item>r2c3</item>
    </SOAP-ENC:Array>

    雖然上面的例子把數(shù)組編碼為獨立的元素,但元素值也可以是嵌入形式,而且若元素值是single reference時,必須編碼為嵌入形式。下例是一個schema片斷,電話號碼數(shù)組嵌入到一個類型為"Person"的結構中,并且通過accessor "phone-numbers"訪問它:

    <simpleType name="phoneNumber" base="string"/>
    <element name="ArrayOfPhoneNumbers">
    <complexType base="SOAP-ENC:Array">
    <element name="phoneNumber" type="tns:phoneNumber" maxOccurs="unbounded"/>
    </complexType>
    <anyAttribute/>
    </element>
    <element name="Person">
    <complexType>
    <element name="name" type="string"/>
    <element name="phoneNumbers" type="tns:ArrayOfPhoneNumbers"/>
    </complexType>
    </element>
    <xyz:Person>
    <name>John Hancock</name>
    <phoneNumbers SOAP-ENC:arrayType="xyz:phoneNumber[2]">
    <phoneNumber>206-555-1212</phoneNumber>
    <phoneNumber>1-888-123-4567</phoneNumber>
    </phoneNumbers>
    </xyz:Person>

    下面的例子中,數(shù)組值為single-reference,被編碼為嵌入元素,包含它的元素名即為入口名:

    <xyz:PurchaseOrder>
    <CustomerName>Henry Ford</CustomerName>
    <ShipTo>
    <Street>5th Ave</Street>
    <City>New York</City>
    <State>NY</State>
    <Zip>10010</Zip>
    </ShipTo>
    <PurchaseLineItems SOAP-ENC:arrayType="Order[2]">
    <Order>
    <Product>Apple</Product>
    <Price>1.56</Price>
    </Order>
    <Order>
    <Product>Peach</Product>
    <Price>1.48</Price>
    </Order>
    </PurchaseLineItems>
    </xyz:PurchaseOrder>

    5.4.2.1 部分儲值(partially transmitted)數(shù)組

    SOAP提供了對部分儲值(partially transmitted)數(shù)組的支持,如某些上下文中的可變數(shù)組。一個partially transmitted 數(shù)組由一個"SOAP-ENC:offset"屬性(從第一個transmitted的元素開始的偏移量,基于0)指示。如果省略,偏移量取0。下面的例子中數(shù)組的大小為5,但只有從0起,第三和第四個元素被儲值。

    <SOAP-ENC:Array ;SOAP-ENC:arrayType="xsd:string[5]" ;SOAP-ENC:offset="[2]">
    <item>The third element</item>
    <item>The fourth element</item>
    </SOAP-ENC:Array>

    5.4.2.2 稀疏數(shù)組Sparse Arrays

    SOAP提供了對稀疏數(shù)組的支持。每個表示成員值的元素包含一個"SOAP-ENC:position"屬性,用來指示它在數(shù)組中的位置。下例是兩維字符串稀疏數(shù)組的例子,數(shù)組大小是4,但只用到第2個。

    <SOAP-ENC:Array SOAP-ENC:arrayType="xsd:string[,][4]">
    <SOAP-ENC:Array href="#array-1" SOAP-ENC:position="[2]"/>
    </SOAP-ENC:Array>
    <SOAP-ENC:Array id="array-1" SOAP-ENC:arrayType="xsd:string[10,10]">
    <item SOAP-ENC:position="[2,2]">Third row, third col</item>
    <item SOAP-ENC:position="[7,2]">Eighth row, third col</item>
    </SOAP-ENC:Array>

    如果對array-1的引用僅發(fā)生在數(shù)組內(nèi)部,上例也可以編碼如下:

    <SOAP-ENC:Array SOAP-ENC:arrayType="xsd:string[,][4]">
    <SOAP-ENC:Array SOAP-ENC:position="[2]" SOAP-ENC:arrayType="xsd:string[10, 10]>
    <item SOAP-ENC:position="[2,2]">Third row, third col</item>
    <item SOAP-ENC:position="[7,2]">Eighth row, third col</item>
    </SOAP-ENC:Array>
    </SOAP-ENC:Array>

    5.4.3 一般復合類型

    在這里提到的編碼規(guī)則不僅僅限于accessor名已知的情況,如果accessor名是運行環(huán)境下實時獲得的,編碼規(guī)則同樣適用,也就是說accessor編碼成一個元素名與accessor名匹配的元素,同時accessor可能包含或者引用該元素的值。如果accessor包含類型不能事先確定的值,它必須包含一個合適的屬性xsi:type 。類似地,上述引用的規(guī)則已經(jīng)足夠用于復合類型的序列化,這些復合類型可能包含用名區(qū)分的accessors(結構)和用名及序數(shù)位置區(qū)分的accessors。(可能包含重復的accessor) 實際上這并不要求任何schema模式包含這些類型,但更為準確的說法是:一個類型模型(type-model)schema如果有這些類型,就可以構造一個符合XML句法規(guī)則的schema和XML文檔實例。

    <xyz:PurchaseOrder>
    <CustomerName>Henry Ford</CustomerName>
    <ShipTo>
    <Street>5th Ave</Street>
    <City>New York</City>
    <State>NY</State>
    <Zip>10010</Zip>
    </ShipTo>
    <PurchaseLineItems>
    <Order>
    <Product>Apple</Product>
    <Price>1.56</Price>
    </Order>
    <Order>
    <Product>Peach</Product>
    <Price>1.48</Price>
    </Order>
    </PurchaseLineItems>
    </xyz:PurchaseOrder>

    類似地,將一個結構上類似數(shù)組但實際上不是一個 SOAP-ENC:Array類型或者 SOAP-ENC:Array子類型的復合值序列化同樣是允許的,例如:

    <PurchaseLineItems>
    <Order>
    <Product>Apple</Product>
    <Price>1.56</Price>
    </Order>
    <Order>
    <Product>Peach</Product>
    <Price>1.48</Price>
    </Order>
    </PurchaseLineItems>

    5.5 缺省值

    省略accessor元素意味著或者有一個缺省值或者值不知道。具體細節(jié)依靠這個accessor,方法和上下文。例如,對于多態(tài)accessor,省略accessor一般意味著一個Null值。同樣,省略布爾accessor一般意味著False值或者值不知道,省略數(shù)字accessor一般意味著值為零或者值不知道。

    5.6 SOAP root屬性

    SOAP root 屬性可用于標記一個序列化root,從而一個對象可以反序列化(deserialized),而實際上該root并不是真正的對象root。這個屬性有兩個可選值"1" or "0"。對象真正的roots屬性值為“1” ,序列化root但不是真正的root屬性值也為“1”,元素如果要顯式地指定不能為序列化root,只需將該屬性設置為“0” SOAP root屬性可以出現(xiàn)在SOAP頭和SOAP體元素的任意子元素中。(譯者注:SOAP root屬性為0的元素不是一個獨立的實體,外部的應用不能訪問到該元素,但該元素可以被SOAP文檔本身的其它元素訪問到)SOAP root屬性可以出現(xiàn)在SOAP頭和SOAP體元素的任意子元素中。這個屬性沒有缺省值。

    6. 在HTTP中使用SOAP

    這一節(jié)講述了如何在HTTP中使用SOAP。把SOAP綁定到HTTP,無論使用或不用HTTP擴展框架,都有很大的好處:在利用SOAP的形式化和靈活性的同時,使用HTTP種種豐富的特性。在HTTP中攜帶SOAP消息,并不意味著SOAP改寫了HTTP已有的語義,而是將構建在HTTP之上SOAP語義自然地對應到HTTP語義。SOAP自然地遵循HTTP的請求/應答消息模型使得SOAP的請求和應答參數(shù)可以包含在HTTP請求和應答中。注意,SOAP的中間節(jié)點與HTTP的中間節(jié)點并不等同,即,不要期望一個根據(jù)HTTP連接頭中的域尋址到的HTTP中間節(jié)點能夠檢查或處理HTTP請求中的SOAP消息。
    在HTTP消息中包含SOAP實體時,按照RFC2376[3] HTTP應用程序必須使用媒體類型 "text/xml"。

    6.1 SOAP HTTP請求

    雖然SOAP可能與各種HTTP請求方式相結合,但是綁定僅定義了在HTTP POST請求中包含SOAP消息。(第7節(jié)中描述了如何在RPC中使用SOAP,第6.3節(jié)描述了如何使用HTTP擴展框架)

    6.1.1 HTTP頭中SOAPAction域

    一個HTTP請求頭中的SOAPAction域用來指出這是一個SOAP HTTP請求,它的值是所要的URI。在格式、URI的特性和可解析性上沒有任何限制。當HTTP客戶發(fā)出SOAP HTTP請求時必須使用在HTTP頭中使用這個域。

    soapaction = "SOAPAction" ":" [ <"> URI-reference <"> ]
    URI-reference = <as defined in RFC 2396 [4]>

    HTTP頭中SOAPAction域使服務器(如防火墻)能正確的過濾HTTP中SOAP請求消息。如果這個域的值是空字符串(""),表示SOAP消息的目標就是HTTP請求的URI。這個域沒有值表示沒有SOAP消息的目標的信息。例子:

    SOAPAction: "http://electrocommerce.org/abc#MyMessage"
    SOAPAction: "myapp.sdl"
    SOAPAction: ""
    SOAPAction:

    6.2 SOAP HTTP應答

    SOAP HTTP遵循HTTP 中表示通信狀態(tài)信息的HTTP狀態(tài)碼的語義。例如,2xx狀態(tài)碼表示這個包含了SOAP組件的客戶請求已經(jīng)被成功的收到,理解和接受。在處理請求時如果發(fā)生錯誤,SOAP HTTP服務器必須發(fā)出應答HTTP 500 "Internal Server Error",并在這個應答中包含一個SOAP Fault元素(見4.4節(jié))表示這個SOAP處理錯誤。

    6.3 HTTP擴展框架

    一個SOAP消息可以與HTTP擴展框架 [6]一起使用以區(qū)分是否有SOAP HTTP請求和它的目標。是使用擴展框架或是普通的HTTP關系到通信各方的策略和能力。通過使用一個必需的擴展聲明和"M-"HTTP方法名前綴,客戶可以強制使用HTTP擴展框架。服務器可以使用HTTP狀態(tài)碼510 "Not Extended"強制使用HTTP擴展框架。也就是說,使用一個額外的來回,任何一方都可以發(fā)現(xiàn)另一方的策略并依照執(zhí)行。用來表示SOAP使用了擴展框架的擴展標志符是:http://schemas.xmlsoap.org/soap/envelope/

    6.4 SOAP HTTP舉例

    例3 使用POST的SOAP HTTP

    POST /StockQuote HTTP/1.1
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    SOAPAction: "http://electrocommerce.org/abc#MyMessage"
    <SOAP-ENV:Envelope...
    HTTP/1.1 200 OK
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope...

    例4 使用擴展框架的SOAP HTTP

    M-POST /StockQuote HTTP/1.1
    Man: "http://schemas.xmlsoap.org/soap/envelope/"; ns=NNNN
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    NNNN-SOAPAction: "http://electrocommerce.org/abc#MyMessage"
    <SOAP-ENV:Envelope...
    HTTP/1.1 200 OK

    Ext:
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope...

    7. 在RPC中使用SOAP

    設計SOAP的目的之一就是利用XML的擴展性和靈活性來封裝和交換RPC調(diào)用。這一節(jié)定義了遠程過程調(diào)用和應答的統(tǒng)一表示形式。雖然可以預計到這種表示形式最可能被用于與第5節(jié)中定義的編碼方式相結合,但也可能有其它的表示形式。SOAP的encodingStyle屬性(見4.3.2節(jié))可以用來表明方法調(diào)用和應答都使用這一節(jié)所指定的表示方式。在RPC中使用SOAP和SOAP協(xié)議綁定(見第6節(jié))是緊密相關的。在使用HTTP作為綁定協(xié)議時,一個RPC調(diào)用自然地映射到一個HTTP請求,RPC應答同樣映射到HTTP應答。但是,在RPC中使用SOAP并不限于綁定HTTP協(xié)議。
    要進行方法調(diào)用,以下的信息是必需的:

    • 目標對象的URI
    • 方法名
    • 方法signature(可選)
    • 方法的參數(shù)
    • 頭數(shù)據(jù)(可選)

    SOAP依靠協(xié)議綁定提供傳送URI的機制。例如,對HTTP來說,請求的URI指出了調(diào)用的來源 。除了必須是一個合法的URI之外,SOAP對一個地址的格式?jīng)]有任何限制。(更多URI的信息參見 [4])

    7.1 RPC和SOAP體

    RPC方法調(diào)用和應答都包含在SOAP Body元素中(見4.3節(jié)),它們使用如下的表示形式:

    • 一個方法調(diào)用用一個結構表示
    • 一個方法調(diào)用被看作一個單個的結構,每個[in]和[in/out]參數(shù)有一個accessor。結構的名和類型與方法相同。每個[in]和[in/out]參數(shù)都被看作一個accessor,這個accessor的名和類型與參數(shù)的名和類型相對應。它們的出現(xiàn)順序和方法中定義的參數(shù)順序相同。
    • 一個方法應答用一個結構表示。
    • 一個方法應答被看作一個單個的結構,返回值和每個[in]和[in/out]參數(shù)有一個accessor。第一個accessor是返回值,之后是參數(shù)accessor,參數(shù)accessor的出現(xiàn)順序和方法中定義的參數(shù)順序相同。每個參數(shù)accessor的名稱和類型與參數(shù)的名稱和類型相對應。返回值accessor的名稱并不重要。同樣,結構的名稱也不重要,不過,通常在方法名稱的后面加上字符串"Response"作為結構的名稱。

    方法錯誤使用SOAP Fault元素(見4.4節(jié))表示。如果綁定的協(xié)議有額外的規(guī)則表示錯誤,則這些規(guī)則也必須要遵從。正如上面所述,方法調(diào)用和應答結構可以按照第5節(jié)中規(guī)則編碼,或者用encodingStyle屬性(見4.1.1節(jié))指定編碼方式。應用程序可以處理缺少參數(shù)的請求,但是可能返回一個錯誤。因為返回結果表示調(diào)用成功,錯誤表示調(diào)用失敗,所以,在方法應答中同時包含返回結果和錯誤是錯誤的。

    7.2 RPC和SOAP頭

    在RPC編碼中,可能會有與方法請求有關但不是正規(guī)的方法signature的附加信息。如果這樣,它必須作為SOAP頭元素的子元素。使用這種頭元素的一個例子是在消息中傳遞事務ID。由于事務ID不是方法signature的一部分,通常由底層的組件而不是應用程序代碼控制,所以沒有一種直接的方法在調(diào)用中傳遞這個必要的信息。通過在頭中添加一個給定名字的條目,接收方的事務管理器就可以析取這個事務ID,而且不影響遠程過程調(diào)用的代碼。

    8. 安全性考慮

    這篇文檔中沒有涉及完整性和保密性,這些問題將在以后的版本中描述。

    9. 參考文獻

    [1] S. Bradner, "The Internet Standards Process -- Revision 3", RFC2026, Harvard University, October 1996
    [2] S. Bradner, "Key words for use in RFCs to Indicate Requirement Levels", RFC 2119, Harvard University, March 1997
    [3] E. Whitehead, M. Murata, "XML Media Types", RFC2376, UC Irvine, Fuji Xerox Info. Systems, July 1998
    [4] T. Berners-Lee, R. Fielding, L. Masinter, "Uniform Resource Identifiers (URI): Generic Syntax", RFC 2396, MIT/LCS, U.C.Irvine, Xerox Corporation, A ugust 1998.
    [5] R. Fielding, J. Gettys, J. C. Mogul, H. Frystyk, T. Berners-Lee, "Hypert ext Transfer Protocol -- HTTP/1.1", RFC 2616, U.C. Irvine, DEC W3C/MIT, DEC,W3C/MIT, W3C/MIT, January 1997
    [6] H. Nielsen, P. Leach, S. Lawrence, "An HTTP Extension Framework", RFC 2774, Microsoft, Microsoft, Agranat Systems
    [7] W3C Recommendation "The XML Specification"
    [8] W3C Recommendation "Namespaces in XML"
    [9] W3C Working Draft "XML Linking Language". This is work in progress.
    [10] W3C Working Draft "XML Schema Part 1: Structures". This is work in progress.
    [11] W3C Working Draft "XML Schema Part 2: Datatypes". This is work in progress.
    [12] Transfer Syntax NDR, in "DCE 1.1: Remote Procedure Call"
    [13] N. Freed, N. Borenstein, "Multipurpose Internet Mail Extensions (MIME)Part One: Format of Internet Message Bodies", RFC2045, Innosoft, First Virtu al, November 1996

    10。 附錄

    A. SOAP封裝舉例

    A.1 請求編碼舉例

    例5 類似于例1,但有一個必要的頭

    POST /StockQuote HTTP/1.1
    Host: www.stockquoteserver.com
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    SOAPAction: "Some-URI"
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"/>
    <SOAP-ENV:Header>
    <t:Transaction
    xmlns:t="some-URI"
    SOAP-ENV:mustUnderstand="1">
    5
    </t:Transaction>
    </SOAP-ENV:Header>
    <SOAP-ENV:Body>
    <m:GetLastTradePrice xmlns:m="Some-URI">
    <symbol>DEF</symbol>
    </m:GetLastTradePrice>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    例6 類似于例1,但有多個請求參數(shù)

    POST /StockQuote HTTP/1.1
    Host: www.stockquoteserver.com
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    SOAPAction: "Some-URI"
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"/>
    <SOAP-ENV:Body>
    <m:GetLastTradePriceDetailed
    xmlns:m="Some-URI">
    <Symbol>DEF</Symbol>
    <Company>DEF Corp</Company>
    <Price>34.1</Price>
    </m:GetLastTradePriceDetailed>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    A.2 應答編碼舉例

    例7 與例2類似,但有必要的頭部

    HTTP/1.1 200 OK
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"/>
    <SOAP-ENV:Header>
    <t:Transaction xmlns:t="some-URI" xsi:type="xsd:int" mustUnderstand="1"> 5 </t:Transaction>
    </SOAP-ENV:Header>
    <SOAP-ENV:Body>
    <m:GetLastTradePriceResponse xmlns:m="Some-URI">
    <Price>34.5</Price>
    </m:GetLastTradePriceResponse>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    例8 與例2類似,但有一個結構

    HTTP/1.1 200 OK
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
    SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"/>
    <SOAP-ENV:Body>
    <m:GetLastTradePriceResponse
    xmlns:m="Some-URI">
    <PriceAndVolume>
    <LastTradePrice> 34.5 </LastTradePrice>
    <DayVolume> 10000 </DayVolume>
    </PriceAndVolume>
    </m:GetLastTradePriceResponse>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    例9 與例2類似,但處理必要的頭出錯

    HTTP/1.1 500 Internal Server Error
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/">
    <SOAP-ENV:Body>
    <SOAP-ENV:Fault>
    <faultcode>SOAP-ENV:MustUnderstand</faultcode>
    <faultstring>SOAP Must Understand Error</faultstring>
    </SOAP-ENV:Fault>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    例10 與例2類似,但處理Body出錯

    HTTP/1.1 500 Internal Server Error
    Content-Type: text/xml; charset="utf-8"
    Content-Length: nnnn
    <SOAP-ENV:Envelope
    xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/">
    <SOAP-ENV:Body>
    <SOAP-ENV:Fault>
    <faultcode>SOAP-ENV:Server</faultcode>
    <faultstring>Server Error</faultstring>
    <detail>
    <e:myfaultdetails xmlns:e="Some-URI">
    <message>
    My application didn't work
    </message>
    <errorcode> 1001 </errorcode>
    </e:myfaultdetails>
    </detail>
    </SOAP-ENV:Fault>
    </SOAP-ENV:Body>
    </SOAP-ENV:Envelope>

    posted on 2007-11-09 22:11 禮物 閱讀(1750) 評論(0)  編輯  收藏 所屬分類: soap
    主站蜘蛛池模板: 色天使亚洲综合在线观看| 亚洲视频国产视频| 成年免费a级毛片| 亚洲AV无码一区二区三区在线观看 | 国产精品色拉拉免费看| 久久精品国产精品亚洲毛片| 日韩精品人妻系列无码专区免费| 亚洲AV日韩AV永久无码下载| 无码成A毛片免费| 亚洲精品亚洲人成在线观看麻豆 | eeuss免费天堂影院| 国产成A人亚洲精V品无码性色| 免费在线观影网站| 久久丫精品国产亚洲av不卡 | 国产精品免费大片一区二区| 亚洲精品无码永久在线观看你懂的| 国产在线观看无码免费视频| 亚洲国产日韩一区高清在线| 国产91色综合久久免费分享| 亚洲精品无码国产片| 免费在线黄色网址| 免费国产成人α片| 亚洲午夜一区二区电影院| 免费观看的毛片手机视频| aa午夜免费剧场| 亚洲综合在线成人一区| 男女交性永久免费视频播放| 国产高潮流白浆喷水免费A片 | 亚洲电影免费在线观看| 国内一级一级毛片a免费| 香蕉国产在线观看免费| 亚洲视频在线观看不卡| 免费国产真实迷j在线观看| 黄网站色视频免费在线观看的a站最新| 91亚洲一区二区在线观看不卡| 在线播放免费人成视频在线观看 | 免费福利资源站在线视频| 亚洲电影免费在线观看| 国产a不卡片精品免费观看| 男人都懂www深夜免费网站| 亚洲中文无码永久免|