本文將討論怎么在J2EE組件中引用Web服務、并且通過JNDI來查找Web服務。在內容組織上,首先把一個EJB2.1無狀態會話Bean發布成Web服務,然后開發基于
jsp的Web服務客戶端,最后著重介紹怎么在JSP的部署時引用Web服務,并且討論怎么通過JNDI來查找和調用Web服務。
閱讀本文前您需要以下的知識和工具:
J2EE1.4 SDK,并且會初步使用;
把握基本的JAX-RPC Web服務開發技能;
了解JNDI的基本知識,能夠使用它進行簡單編程;
一般的Java編程知識。
Web服務客戶端 我們知道,JAX-RPC Web服務客戶端有以下幾個類型:
基于Stub;
基于動態代理;
基于動態調用接口(DII)。
實際上,上面三種客戶端都是使用Service接口來作為它們的創建工廠,Service接口中定義了諸如以下的方法:
例程1 Service接口中的某些方法
Call createCall() ;
Call createCall(QName portName,
String
OperationName) ;
Remote getPort(QName portName,
Class serviceEndpointInterface) ;
Remote getPort(Class serviceEndpointInterface) ;
可以看出,通過Service接口,能夠創建Call對象和Remote對象,而Call或者Remote對象正是調用Web服務所需要的。通過使用Service接口,我們可以在Web服務客戶端通過以下的方式來實現調用:
例程2 在客戶端調用Web服務
//創建一個ServiceFactory對象。
ServiceFactory serviceFactory
= ServiceFactory.newInstance();
//通過ServiceFactory
對象創建一個調用Web服務的Service對象。
Service service =
serviceFactory.createService
(taxWsdlUrl,
new QName(nameSpaceUri,
serviceName));
//獲得服務端點實例。
TaxService my
PRoxy =
( TaxService) service.getPort
(new QName(nameSpaceUri,
portName),
TaxService.class);
//調用Web服務。
double result=myProxy.calculateTax(5000);
可以看出,在創建Service實例時,需要使用指定的WSDL文件位置、由服務名和名稱空間URI組成的有效名稱空間,這樣使得創建這個實例時變得復雜。JAX-RPC規范推薦使用JNDI來查找服務接口。通過JNDI,使得調用Web服務時就像調用EJB一樣簡單。只要通過兩步就可以獲得Web服務接口:
初始化一個名稱空間上下文;
在這個上下文中查找Web服務。
比如可以按照以下的方式來進行:
例程3 通過JNDI調用Web服務 1
InitialContext ic = new InitialContext ();
Service abf = (Service)ic.lookup( "java:comp/env/service/AddressBookService");
Web服務引用的名字(AddressBookService)在部署時指定,java:comp/env是JNDI的上下文,service是Web服務的sub context。所以Web服務的JNDI名字一般由以下幾個部分組成:
Web服務JNDI=客戶端環境上下文+
service(sub context)+服務引用名字
我們看到,例程3中查找到的是Service接口,其實在開發中,我們可以采用另一種服務引用形式:直接查找Web服務接口,如例程4所示。
例程4通過JNDI調用Web服務2
Context ic= new InitialContext();
HelloServiceInterface service =
(HelloServiceInterface) ic.lookup
("java:comp/env/service/HelloService");
在后面的例子可以看到,使用這個方式在調用時又簡化了一步。下面我們通過一個實例來演示怎么在J2EE Web服務的客戶端引用Web服務,然后通過JNDI來查找Web服務。
開發、部署一個Web服務
我們開發一個提供個人所得稅計算的Web服務,采用EJB作為服務端點。首先定義一個接口,如例程5所示。
例程5 定義服務接口
package com.hellking.
study.webservice.tax;
import java.rmi.Remote;
import java.rmi.RemoteException;
/**
*個人所得稅Web服務。
*/
public interface TaxService
extends Remote
{
public double calculateTax
(double salary)throws
java.rmi.RemoteException;
}
它提供的服務方法是計算個人所得稅。
例程6 EJB部分代碼
double base=1200;
//個人所得稅基數,
2003年10起北京為1200元。
//業務邏輯代碼,
實現服務端點接口中定義的方法。
public double calculateTax(double salary)
{
return getTax(salary-base);
}
//下面是具體的計算方法。公式適合于現在的個人所得稅制度。
private double getTax(double tax_salary)
{
double tax=0.0d;
if(0>tax_salary) tax=0;
else if(0<tax_salary&&tax_salary <=500)
tax=tax_salary*0.05-0;
else if(500<tax_salary&&tax_salary<=2000)
tax=tax_salary*0.10-25;
else if(2000<tax_salary&&tax_salary<=5000)
tax=tax_salary*0.15-125;
else if(5000<tax_salary&&tax_salary<=20000)
tax=tax_salary*0.20-375;
else if(20000<tax_salary&&tax_salary<=40000)
tax=tax_salary*0.25-1375;
else if(40000<tax_salary&&tax_salary<=60000)
tax=tax_salary*0.30-3375;
else if(60000<tax_salary&&tax_salary<=80000)
tax=tax_salary*0.35-6375;
else if(80000<tax_salary&&tax_salary<=100000)
tax=tax_salary*0.40-10375;
else if(100000<tax_salary)
tax=tax_salary*0.45-15375;
return tax;
}
下面編寫一個配置文件,通過配置文件來生成WSDL到JAX-RPC之間的映射描述符。配置文件如下:
<?xml version="1.0"
encoding="UTF-8"?>
<configuration
xmlns="http://java.sun.com
/xml/ns/jax-rpc/ri/config">
<service
name="MyTaxService"
targetNamespace="urn:Tax"
typeNamespace="urn:Tax"
packageName="com.hellking.
study.webservice.tax">
<interface name="com.hellking.
study.webservice.tax.TaxService"/>
</service>
</configuration>
注重這里Web服務的名字是MyTaxService,名稱空間是"urn:Tax",服務接口是"com.hellking.study.webservice.tax.TaxService",這些參數將在后面的編程中使用。通過以下命令來生成一個mapping.xml映射文件:
wscompile -define -d . -nd .
-classpath . -mapping mapping.xml
config.xml
發客戶端
這個例子提供了兩種不同引用Web服務的方法,如例程8所示。
例程8 在客戶端通過JNDI查找Web服務
package com.hellking.study.webservice.tax;
import javax.naming.*;
import javax.xml.rpc.Service;
import javax.xml.namespace.QName;
/**
*Web服務客戶演示:通過JNDI來查找Web服務。
*/
public class TaxBean
{
/**
*第一種查找服務的方法,
直接獲得MyTaxService接口。
*/
public double getTax1(double sal)
{
double ret=0;
try
{
Context ctx=new InitialContext();
MyTaxService taxService
=(MyTaxService)ctx.lookup
("java:comp/env/service/tax");
//通過MyTaxService獲得TaxService服務端點接口。
TaxService
tax=taxService.getTaxServicePort();
ret=tax.calculateTax(sal);
}
catch(Exception e)
{
System.out.println(e);
}
return ret;
}
/**
*另一種查找服務的方法,
獲得的是Service接口,
然后再通過這個接口來獲得具體的服務。
*/
public double getTax2(double sal)
{
double ret=0;
try
{
Context ctx=new InitialContext();
Service service=
(Service)ctx.lookup
("java:comp/env/service/tax2");
QName portQName= new QName
("urn:Tax","TaxService");
//使用這種方式獲得服務端點接口時,
需要指定名稱空間。
TaxService tax=(TaxService)
service.getPort(portQName,
com.hellking.study.
webservice.tax.TaxService.class);
ret=tax.calculateTax(sal);
}
catch(Exception e)
{
e.printStackTrace();
System.out.println(e);
}
return ret;
}
}
可以看出,第一種方法查找的就是MyTaxService接口,而第二種方法查找的是Service接口。具體使用那種方式,是和部署描述相關的,在后面將介紹部署的差別。最后開發一個JSP來作為測試客戶端,這個JSP通過JavaBean調用Web服務,如例程9所示。
例程9 測試的JSP
<%@ page contentType="text/Html;charset=gb2312"%>
<%@ page import="com.hellking.study.webservice.tax.*,javax.naming.*"%>
<jsp:useBean id="tax" class="com.hellking.study.webservice.tax.TaxBean"/>
<%
double salary=0;
try{
salary=Double.parseDouble
((String)request.getParameter
("salary"));
}
catch(Exception e){}
%>
<html>
<head>
<title>通過JNDI調用Web服務。</title>
</head>
<body>
<div align="center">
<h1>Web服務----適合北京地區,
2003年</h1>
<%
out.println("個人所得稅是:<br>");
out.println(tax.getTax1(salary));
out.println("<br>另一種方法調用Web服務,
個人所得稅是:<br>");
out.println(tax.getTax2(salary));
%>
<hr>
<form action="/tax/tax">
<table border=1>
<tr bgcolor=654321>
<td >輸入工資</td>
<td><input type=text name=salary></td>
</tr>
<tr><td colspan=2><input
type=submit value=查看></td>
</tr>
</table>
</form>
</div>
<hr>
</body>
</html>
最后來看具體的部署描述符。
在客戶端的描述中引用Web服務
打開J2EESDK部署工具(執行%J2EESDK_HOME%//AppServer//bin//deploytool.bat或者$J2EESDK/AppServer/bin/deploytool.sh),新建一個Web應用,把上面的測試JSP添加進來。在部署時,Web應用將包含如圖1所示的文件。

圖1 Web應用中包含的文件
點擊這個Web應用,在右邊再點擊【Web services Refs】選項卡,再點擊【Add】按鈕。現在可以增加Web服務引用了。增加一個名稱為service/tax的Web服務引用,如圖2所示。
點擊查看大圖圖2 增加Web服務引用
注重上面的Service接口是com.hellking.study.webservice.tax.MyTaxService。然后點擊【Container Managed Ports】,如圖3所示。
點擊查看大圖圖3 增加容器治理端點
到此,一種引用Web服務的方式就完成了。
下面看服務接口直接是Service的情況,再增加一個Web服務引用,如圖4所示。
點擊查看大圖圖4 增加另一種服務引用
注重上面Service Interface是javax.xml.rpc.Service,并且指定了名稱空間(urn:Tax)和Local Part(MyTaxService)。同樣,按照圖3所示方法增加一個容器治理端點,端點接口名稱和端口組件名稱和圖3一致。
經過了上面的部署,實際上在web.xml中生成了以下的部署描述符。
例程 10 生成的部署描述符
<service-ref>
<service-ref-name>
service/tax</service-ref-name>
<service-interface>com.hellking.
study.webservice.tax.
MyTaxService</service-interface>
<wsdl-file>WEB-INF/wsdl
/MyTaxService.wsdl</wsdl-file>
<jaxrpc-mapping-file>mapping.xml
</jaxrpc-mapping-file>
<port-component-ref>
<service-endpoint-interface>
com.hellking.study.
webservice.tax.TaxService
</service-endpoint-interface>
<port-component-link>
TaxServicePort</port-component-link>
</port-component-ref>
</service-ref>
<service-ref>
<service-ref-name>
service/tax2</service-ref-name>
<service-interface>
javax.xml.rpc.Service</service-interface>
<wsdl-file>WEB-INF/wsdl
MyTaxService.wsdl</wsdl-file>
<jaxrpc-mapping-file>mapping.xml
</jaxrpc-mapping-file>
<service-qname
xmlns:service-qname_ns__=
"urn:Tax">service-qname_ns__:MyTaxService
</service-qname>
<port-component-ref>
<service-endpoint-interface>
com.hellking.study.webservice.tax.TaxService
</service-endpoint-interface>
<port-component-link>
TaxServicePort</port-component-link>
</port-component-ref>
</service-ref>
下面解釋一下這個描述付。對Web服務的引用通過元素來指定;就是在客戶端編程中要使用的引用名字;就是服務接口,有兩種,分別是javax.xml.rpc.Service和com.hellking.study.webservice.tax.MyTaxService;是服務的有效名稱空間,假如直接使用com.hellking.study.webservice.tax.MyTaxService作為服務接口,就不需要指定元素;就是對服務端點的引用,它引用的是webservices.xml中定義元素。用來鏈接到webservices.xml中指定的,并且兩者的名字是一致的。
調用測試
部署完成后,在瀏覽器里輸入:
http://127.0.0.1:8080/tax/tax
將出現如圖5所示的界面。

圖5 調用Web服務
總結
通過上面的介紹,相信讀者對Web服務的引用已經有了全面的熟悉。我們可以看到,通過使用JNDI,在J2EE組件中調用Web服務就像面向對象編程一樣,甚至可以不理解WSDL、XML之類的概念就能調用Web服務。