Web服務(wù)應(yīng)用程序存在的一個(gè)常見問題是,真正的后端軟件不支持XML(或者至少不支持你Web服務(wù)所使用的標(biāo)準(zhǔn)化的XML)。要解決這個(gè)問題,很多架構(gòu)都實(shí)施了一個(gè)翻譯階段,用來解釋收到的XML,并將它轉(zhuǎn)換成為后端系統(tǒng)能夠識(shí)別的更加適當(dāng)?shù)母袷?。雖然將XML翻譯成不同的格式有很多種方式法,但是可擴(kuò)展樣式表語言(eXtensibleStylesheet Language,XSL)提供了一個(gè)強(qiáng)健的、標(biāo)準(zhǔn)的和對(duì)XML友好的解決方案。
一些常見的問題
在XML消息和將要處理它的應(yīng)用程序之間,存在著很多種可能需要翻譯的地方。但是,這也存在著很多常見的問題,包括:
查詢(Lookup)
映射(Mapping)
聚合(Aggregation)
分割(Splitting)
公式(Formulas)
重排序(Reordering)
查詢處理會(huì)接受收到的值,并為目標(biāo)系統(tǒng)將它映射到不同的值。例如,你的XML有一個(gè)值是309,但是應(yīng)用程序需要的值是“Uber Widget”。
映射在本質(zhì)上是將來自一個(gè)字段的值重新分配為另一個(gè)值。例如,在XML里,你可能有一個(gè)AccountNumber元素,它需要被重新分配成為一個(gè)叫做CustomerAccountNumber的新元素。
聚合處理是將兩個(gè)或者更多來自XML的項(xiàng)目組合成一個(gè)用于后端系統(tǒng)的單個(gè)項(xiàng)目。一個(gè)常見的例子是將姓和名字段組合成單一的姓名(字段)。分割是聚合的逆向操作,它是將XML的一個(gè)值分成兩個(gè)或者更多分開的組件。
公式處理一般涉及對(duì)一個(gè)或者更多XML值的計(jì)算,以獲得一個(gè)用于新應(yīng)用程序的值。一個(gè)例子是利用XML里子定單來計(jì)算定單的總量。
最后,重排序是更改XML里項(xiàng)目的序列或者結(jié)構(gòu)的處理,這樣它們就能夠符合目標(biāo)系統(tǒng)所期望的序列或者結(jié)構(gòu)。
映射舉例
現(xiàn)在讓我們用一個(gè)例子來更加仔細(xì)地看看這個(gè)過程。我們假設(shè)收到的XML就像Listing 1里的一樣:
Listing 1: webserviceorder.xml
?xml version="1.0" ?>
Order>
OrderNumber>8100/OrderNumber>
AccountNumber>99213/AccountNumber>
Item>
SKU>2388/SKU>
Description>Uber Widget/Description>
Quantity>15/Quantity>
PricePer>10.95/PricePer>
/Item>
Item>
SKU>6273/SKU>
Description>Flangeoid/Description>
Quantity>10/Quantity>
PricePer>52.00/PricePer>
/Item>
/Order>
現(xiàn)在,我們的訂單系統(tǒng)需要一個(gè)稍有不同的格式。我們所需要做的是將從Web服務(wù)收到的訂單轉(zhuǎn)換成為L(zhǎng)isting 2里的這種格式。
Listing 2: applicationorder.xml
?xml version="1.0" encoding="UTF-8" ?>
NewOrder>
CustomerAccountNumber>99213/CustomerAccountNumber>
CustomerOrderNumber>8100/CustomerOrderNumber>
OrderItems>
OrderItem>
SKU>2388/SKU>
CustomerPrice>10.95/CustomerPrice>
Quantity>15/Quantity>
Subtotal>164.25/Subtotal>
/OrderItem>
OrderItem>
SKU>6273/SKU>
CustomerPrice>52.00/CustomerPrice>
Quantity>10/Quantity>
Subtotal>520/Subtotal>
/OrderItem>
/OrderItems>
/NewOrder>
翻譯
由于這是一個(gè)經(jīng)過簡(jiǎn)化的例子,所以我們?cè)谵D(zhuǎn)換格式的時(shí)候需要對(duì)XSL模板所做的只有不多的一些事情。我們要注意的第一件事是:我們必須映射一些元素:
把Order映射成NewOrder
把AccountNumber映射成CustomerAccountNumber
把OrderNumber映射成CustomerOrderNumber
把Item映射成OrderItem
把PricePer映射成CustomerPrice
然后,我們需要將OrderItem元素重新安排到一個(gè)叫做OrderItems的新元素之下。最后,我們加入一個(gè)叫做Subtotal的新元素,它會(huì)根據(jù)(商品)單價(jià)和數(shù)量進(jìn)行計(jì)算。
映射翻譯是最簡(jiǎn)單的,因?yàn)槟阒恍枰?jiǎn)單地在模板里定義新元素,并指明這個(gè)新元素?fù)碛衼碜允盏降腦ML文檔里元素的值。對(duì)Items的重排序是通過將Item子模板調(diào)用放進(jìn)一個(gè)叫做OrderItems的新元素里實(shí)現(xiàn)的。最后,使用一個(gè)簡(jiǎn)單的XPath表達(dá)式來進(jìn)行計(jì)算。Listing 3顯示了用來將收到的XML轉(zhuǎn)換成針對(duì)應(yīng)用程序的XML的XSL文檔。
Listing 3: translate.xsl
?xml version="1.0" ?>
xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
xsl:template match="/">
xsl:apply-templates select="Order" />
/xsl:template>
xsl:template match="Order">
NewOrder>
CustomerAccountNumber>xsl:value-of select="AccountNumber" />/CustomerAccountNumber>
CustomerOrderNumber>xsl:value-of select="OrderNumber" />/CustomerOrderNumber>
OrderItems>
xsl:apply-templates select="Item" />
/OrderItems>
/NewOrder>
/xsl:template>
xsl:template match="Item">
OrderItem>
SKU>xsl:value-of select="SKU" />/SKU>
CustomerPrice>xsl:value-of select="PricePer" />/CustomerPrice>
Quantity>xsl:value-of select="Quantity" />/Quantity>
Subtotal>xsl:value-of select="PricePer * Quantity" />/Subtotal>
/OrderItem>
/xsl:template>
/xsl:stylesheet>
本文作者Brian Schaffner是富士通咨詢公司的副主任。他為富士通的技術(shù)咨詢公司提供架構(gòu)、設(shè)計(jì)和開發(fā)支持。