WebService是一種跨編程語言和跨操作系統平臺的遠程調用技術
所謂遠程調用,就是一臺計算機a上的一個程序可以調用到另外一臺計算機b上的一個對象的方法,譬如,銀聯提供給商場的pos刷卡系統(采用交互提問的方式來加深大家對此技術的理解)。
遠程調用技術有什么用呢?商場的POS機轉賬調用的轉賬方法的代碼是在銀行服務器上,還是在商場的pos機上呢?什么情況下可能用到遠程調用技術呢?例如,amazon,天氣預報系統,淘寶網,校內網,百度等把自己的系統服務以webservice服務的形式暴露出來,讓第三方網站和程序可以調用這些服務功能,這樣擴展了自己系統的市場占有率,往大的概念上吹,就是所謂的SOA應用。
所謂跨編程語言和跨操作平臺,就是說服務端程序采用java編寫,客戶端程序則可以采用其他編程語言編寫,反之亦然!跨操作系統平臺則是指服務端程序和客戶端程序可以在不同的操作系統上運行。
除了WebService外,常見的遠程調用技術還有RMI(Remotemethodinvoke)和CORBA,由于WebService的跨平臺和跨編程語言特點,因此比其他兩種技術應用更為廣泛,但性能略低。
使用JDK對Webservice的支持進行Webservice調用時通常的操作步驟如下:
1
2
3
4
5
6
|
//1、創建一個javax.xml.ws.Service實例 javax.xml.ws.Service service = javax.xml.ws.Service.create(wsdl, serviceName); //2、通過Service實例獲取對應的服務接口的代理 HelloService helloService = service.getPort(portName, HelloService. class ); //3、通過獲取到的Webservice服務接口的代理調用對應的服務方法 helloService.sayHello( "Elim" ) |
在上述的步驟一在構建Service實例的同時,在Service內部會構建一個ServiceDelegate類型的對象賦給屬性delegate,內部持有。然后在第二步會利用delegate創建一個服務接口的代理對象,同時還會代理BindingProvider和Closeable接口。然后在第三步真正發起接口請求時,內部會發起一個HTTP請求,發起HTTP請求時會從BindingProvider的getRequestContext()返回結果中獲取超時參數,分別對應com.sun.xml.internal.ws.connection.timeout和com.sun.xml.internal.ws.request.timeout參數,前者是建立連接的超時時間,后者是獲取請求響應的超時時間,單位是毫秒。如果沒有指定對應的超時時間或者指定的超時時間為0都表示永不超時。所以為了指定超時時間我們可以從BindingProvider下手。比如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
public class Client { public static void main(String[] args) throws Exception { String targetNamespace = "http://test.elim.com/ws" ; QName serviceName = new QName(targetNamespace, "helloService" ); QName portName = new QName(targetNamespace, "helloService" ); URL wsdl = new URL( "http://localhost:8888/hello" ); //內部會創建一個ServiceDelegate類型的對象賦給屬性delegate Service service = Service.create(wsdl, serviceName); //會利用delegate創建一個服務接口的代理對象,同時還會代理BindingProvider和Closeable接口。 HelloService helloService = service.getPort(portName, HelloService. class ); BindingProvider bindingProvider = (BindingProvider) helloService; Map<String, Object> requestContext = bindingProvider.getRequestContext(); requestContext.put( "com.sun.xml.internal.ws.connection.timeout" , 10 * 1000 ); //建立連接的超時時間為10秒 requestContext.put( "com.sun.xml.internal.ws.request.timeout" , 15 * 1000 ); //指定請求的響應超時時間為15秒 //在調用接口方法時,內部會發起一個HTTP請求,發起HTTP請求時會從BindingProvider的getRequestContext()返回結果中獲取超時參數, //分別對應com.sun.xml.internal.ws.connection.timeout和com.sun.xml.internal.ws.request.timeout參數, //前者是建立連接的超時時間,后者是獲取請求響應的超時時間,單位是毫秒。如果沒有指定對應的超時時間或者指定的超時時間為0都表示永不超時。 System.out.println(helloService.sayHello( "Elim" )); } } |
完整示例如下:
服務接口:
1
2
3
4
5
6
|
@WebService (portName= "helloService" , serviceName= "helloService" , targetNamespace= "http://test.elim.com/ws" ) public interface HelloService { String sayHello(String name); } |
服務接口實現:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
@WebService (portName= "helloService" , serviceName= "helloService" , targetNamespace= "http://test.elim.com/ws" ) public class HelloServiceImpl implements HelloService { private Random random = new Random(); @Override public String sayHello(String name) { try { TimeUnit.SECONDS.sleep( 5 + random.nextint( 21 )); //隨機睡眠5-25秒 } catch (InterruptedException e) { e.printStackTrace(); } return "Hello " + name; } } |
服務端代碼:
1
2
3
4
5
|
public class Server { public static void main(String[] args) { Endpoint.publish( "http://localhost:8888/hello" , new HelloServiceImpl()); } } |
在上述的服務端代碼中隨機睡眠了5-25秒,而客戶端指定的超時時間是15秒,所以在測試的時候你會看到有時候服務調用會超時,有時會正常響應。
總結
以上就是本文關于Java編程Webservice指定超時時間代碼詳解的全部內容,希望對大家有所幫助。如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
原文鏈接:http://elim.iteye.com/blog/2394410