詳解WebService開發(fā)中四個常見問題
任何問題都需要從它的根源說起,所以簡單說一下WebService的工作原理??蛻舳苏{(diào)用一個WebService的方法,首先需要將方法名和需要傳遞的參數(shù)包裝成XML(也就是SOAP包),通常是通過HTTP傳遞到服務(wù)器端,然后服務(wù)器端解析這段XML,得到被調(diào)用方法名稱和傳遞來的參數(shù),進而調(diào)用WebService實例的相應(yīng)方法。方法執(zhí)行完成之后,將返回的結(jié)果再包裝成XML(SOAP響應(yīng))發(fā)送到客戶端,客戶端解析這段XML,進而得到返回結(jié)果。這里關(guān)鍵的地方在于中間加入了對象和XML相互轉(zhuǎn)換的過程。
問題一:WebService與方法重載
首先說明,WebService不支持方法重載。下面舉例說明。
例如定義如下WebService接口:
1 @WebService |
先來看方法sayHello(),如果客戶端發(fā)送如下SOAP請求:
1 <soap:Envelope> |
從SOAP請求我們可以看出客戶端需要調(diào)用方法為sayHello(),所傳遞的參數(shù)為11,但是無法知道是整數(shù)的11,還是字符串"11",所以也就無法確定所調(diào)用的方法是哪一個。
接下來看一下sayHello2(),如果客戶端傳遞的參數(shù)只包括一個id值,例如:
1 <soap:Envelope> |
還是無法判斷調(diào)用的是哪個方法,因為可以理解為客戶端傳遞的第二個參數(shù)為空(Null)。
通常情況下,在發(fā)布一個含有重載方法的WebService時會有異常發(fā)生,或者當調(diào)用一個方法時,服務(wù)器端報告找不到相對應(yīng)的方法。
問題二:我的數(shù)據(jù)被修改了?
先來看WebService接口:
1 @WebService |
這里需要注意的是WebService的方法sayHello()的參數(shù)是一個接口,而不是一個具體類(例如Aegis綁定就允許直接發(fā)布這樣的WebService)。在客戶端調(diào)用sayHello()的時候傳遞一個Person對象,它實現(xiàn)了IPerson接口。經(jīng)過中間一系列的XML和對象之間的轉(zhuǎn)換過程,服務(wù)器端得到的只是一個實現(xiàn)了IPerson接口的實例,它不一定就是一個Person對象,如果要強制將其轉(zhuǎn)換為Person,就有可能拋出異常。
問題的根源在于Aegis將XML轉(zhuǎn)換為Java對象是通過Proxy或CGlib這類的工具生成一個“代理類”實現(xiàn)IPerson接口,然后創(chuàng)建這個代理類的一個實例,那它肯定不是一個Person了。
#p#
問題三:循環(huán)引用
還是先來看一個例子。下面是WebService的接口:
1 @WebService |
請注意,Teacher和Student是一對多的“雙向”關(guān)系。在這種情況下,我們可以想一下如何將一個Teacher對象轉(zhuǎn)換成一段XML?
您可能想到下面的答案:
1 <teacher> |
看到了吧,XML竟然也會進入“死循環(huán)”。問題的根源在于對象之間的循環(huán)引用。這種問題通常在客戶端發(fā)送WebService請求之前就會拋出異常,因為無法將這個對象轉(zhuǎn)換為可傳輸?shù)腦ML。
問題四:龐然大物
還是先看一個例子,下面是WebService的接口:
1 @WebService |
這個方法接收一個Student數(shù)組,包含成百上千個Student,與上面例子不同的是Student和Teacher現(xiàn)在是多對一的單向關(guān)系,所以不會有“循環(huán)引用”的問題。假設(shè)所有這些Student的Teacher是一個人。我們試著將這個Student數(shù)組對象轉(zhuǎn)換為一段XML,如下:
1 <student> |
問題出來了,看到了沒有,每個Student節(jié)點下面都有一個Teacher節(jié)點,當這段XML被接收方轉(zhuǎn)換為Student數(shù)組時,每個學生都有了一個自己的老師,Teacher對象被復(fù)制了成百上千次,經(jīng)過這么一個轉(zhuǎn)換--傳輸--轉(zhuǎn)換的過程,這個數(shù)組對象真的成了一個“龐然大物”。
問題的根源在于Student和Teacher之間的關(guān)系是多對一,當傳送“多”方時,“一”方有可能會被復(fù)制多次。從而占用大量網(wǎng)絡(luò)傳輸帶寬和內(nèi)存。在這里參數(shù)不一定非要是一個集合或者數(shù)組,例如ObjectA和ObjectB都有一個對ObjectC的引用,經(jīng)過SOAP傳送過后,ObjectC就由一個變成兩個了,分別屬于ObjectA和ObjectB,而不再是共享一個ObjectC了。
【編輯推薦】