WebService
Webservice:跨语言跨平台的远程调用技术
WebService定义: 顾名思义就是基于Web的服务。它使用Web(HTTP)方式,接收和响应外部系统的某种请求。从而实现远程调用。
Webservice理解:我们可以调用互联网上查询天气信息Web服务,然后将它嵌入到我们的程序(C/S或B/S程序)当中来,当用户从我们的网点看到天气信息时,他会认为我们为他提供了很多的信息服务,但其实我们什么也没有做,只是简单了调用了一下服务器上的一段代码而已。WebSerice可以将你的服务(一段代码)发布到互联网上让别人去调用,也可以调用别人机器上发布的WebService,就像使用自己的代码一样.。
1 远程调用的案例(模拟远程调用天气预报服务)
便民查询网站的分析
天气信息必须通过远程调用的方式获得。
远程调用的方法
1.1.1 Socket实现远程调用
1.1.1.1 实现步骤:
服务端:
第一步:创建一个java工程
第二步:使用ServerSocket创建一个服务。需要指定服务的端口号。
第三步:调用accept()方法等待客户端建立连接。
第四步:建立连接后使用输入流读取客户端发送的城市名称
第五步:查询天气信息。模拟。
第六步:使用输出流返回天气信息。
第七步:关闭输入输出流。服务端不关闭连接,等待客户端关闭。
客户端:
第一步:创建一个java工程
第二步:使用Socket类和服务端建立连接。需要指定ip和端口。
第三步:使用输出流发送城市名称
第四步:使用输入流读取服务端返回的结果
第五步:打印结果。
1.1.1.2 代码实现
开发环境:要求jdk1.7以上
开发工具:eclipse indigo
服务端
public class WeatherServer {
public static void main(String[] args) {
try {
// 创建一个Socket服务
// 参数:服务的端口号,建议大一些0-1024是系统留用端口
ServerSocket serverSocket = new ServerSocket(12345);
System.out.println("服务端已启动。。。。。");
while(true) {
// 等待客户端建立连接
// 阻塞的方法,建立连接后向下执行
final Socket socket = serverSocket.accept();
Runnable runnable = new Runnable() {
@Override
public void run() {
DataInputStream inputStream = null;
DataOutputStream outputStream = null;
try {
// 使用输入流读取客户端发送的数据
inputStream = new DataInputStream(socket.getInputStream());
String cityName = inputStream.readUTF();
System.out.println("接收到客户端发送的数据:" + cityName);
// 根据城市查询天气
System.out.println("查询天气。。。。");
Thread.sleep(1000);
String resultString = "阴天转雷阵雨";
outputStream = new DataOutputStream(socket.getOutputStream());
// 返回查询结果
System.out.println("返回查询结果:" + resultString);
outputStream.writeUTF(resultString);
} catch (Exception e) {
// TODO: handle exception
}finally {
// 关闭流
try {
inputStream.close();
outputStream.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
};
//启动线程
new Thread(runnable).start();
}
} catch (Exception e) {
// TODO: handle exception
}
}
}
客户端
public class WeatherClient {
public static void main(String[] args) throws Exception {
while (true) {
// 创建一个socket连接
Socket socket = new Socket("127.0.0.1", 12345);
// 使用输出流发送城市名称
DataOutputStream outputStream = new DataOutputStream(
socket.getOutputStream());
// 发送城市名称
outputStream.writeUTF("北京");
// 接收返回的结果
DataInputStream inputStream = new DataInputStream(
socket.getInputStream());
String resultString = inputStream.readUTF();
System.out.println("天气信息:" + resultString);
// 关闭流
inputStream.close();
outputStream.close();
}
}
}
Socket要求服务端有持续服务能力,而且要实现多线程。编码比较麻烦。
1.1.2 Webservice实现远程调用
1.1.2.1 实现步骤
服务端:
1. 编写SEI(ServiceEndpoint Interface),SEI在WSDL中称为portType,在java中称为接口。
2. 编写sei实现类,需要实现SEI接口,而且还要求在实现类上添加一个@Webservice注解。
3. 发布服务。使用Endpoint的静态方法publish。
1.1.2.2 代码实现
Sei
public interface WeatherInterface {
String queryWeather(String cityName);
}
SEI实现类
@WebService
//加上此注解传输时使用soap1.2版本的协议
@BindingType(javax.xml.ws.soap.SOAPBinding.SOAP12HTTP_BINDING)
public class WeatherInterfaceImpl implements WeatherInterface {
@Override
public String queryWeather(String cityName) {
System.out.println("接收到客户端发送的城市名称:" + cityName);
//查询天气信息
String result = "雷阵雨";
//返回查询结果
return result;
}
}
发布服务
public class WeatherServer {
public static void main(String[] args) {
//发布服务
//第一个参数:服务发布的地址,就是一个url
//第二个参数:SEI实现类对象
Endpoint.publish("http://127.0.0.1:12345/weather", new WeatherInterfaceImpl());
}
}
1.1.2.3 服务启动成功的判断
访问服务发布地址+?wsdl查看服务的wsdl。如果wsdl中有porttype就说明服务发布成功。
1.1.2.4 Wsimport生成客户端调用代码
Wsimport路径就在jdk的bin目录下。
使用方法:
wsimport是jdk自带的webservice客户端工具,可以根据wsdl文档生成客户端调用代码(java代码).当然,无论服务器端的WebService是用什么语言写的,都可以生成调用webservice的客户端代码,服务端通过客户端代码调用webservice。
wsimport.exe位于JAVA_HOME\bin目录下.
常用参数为:
-d<目录> - 将生成.class文件。默认参数。
-s<目录> - 将生成.java文件。
-p<生成的新包名> -将生成的类,放于指定的包下。如果不指定此参数包名就是wsdl命名空间的倒序
(wsdlurl)- http://server:port/service?wsdl,必须的参数。
示例:
C:/>wsimport –s . http://127.0.0.1:1234/weather?wsdl
注意:-s不能分开,-s后面有个小点
1.1.2.5 客户端创建
1.1.2.5.1 实现步骤
利用生成的代码创建客户端。
第一步:创建一个服务视图对象。
第二步:从服务视图获得porttype(SEI)对象。
第三步:调用服务端方法
第四步:打印结果
1.1.2.5.2 代码实现
public class WeatherClient {
public static void main(String[] args) {
//创建服务视图
WeatherInterfaceImplService service = new WeatherInterfaceImplService();
//获得porttype对象
WeatherInterfaceImpl portType = service.getWeatherInterfaceImplPort();
//调用服务端方法
String result = portType.queryWeather("北京");
System.out.println(result);
}
}
1.1.3 Webservice和socket技术对比
Socket是所有通信的基础也是语言个无关平台无关。
Socket使用的是tcp协议,传输效率高。适合传递大数据高并发场景,高并发的情况需要实现多线程并且使用到线程池,编码复杂。Sockt的高并发框架mina。
Socket只是流的传输,传输的格式需要程序员自己定义。
Webservice使用的是soap协议,soap协议基于http协议的应用层协议,本质就是http+xml。Soap协议是w3c标准,传输效率低。使用传输数据不是太大的场合,也是支持高并发的,受限于web容器。支持soap协议和wsdl两者都是国际通用标准,不需要自定义数据格式,只需要面向对象开发。
2 什么是webservice
Webservice 即web服务,它是一种跨编程语言和跨操作系统平台的远程调用技术即跨平台远程调用技术。
JAVA 中共有三种WebService 规范,分别是JAX-WS(JAX-RPC)、JAXM&SAAJ、JAX-RS。
Webservice的三种规范
2.1.1 JAX-WS
JAX-WS 的全称为 Java API for XML-BasedWebservices ,早期的基于SOAP 的JAVA 的Web 服务规范JAX-RPC(Java API For XML-Remote ProcedureCall)目前已经被JAX-WS 规范取代。从java5开始支持JAX-WS2.0版本,Jdk1.6.0_13以后的版本支持2.1版本,jdk1.7支持2.2版本。
采用标准SOAP(SimpleObject Access Protocol) 协议传输,soap属于w3c标准。Soap协议是基于http的应用层协议,soap协议传输是xml数据。
采用wsdl作为描述语言即webservice使用说明书,wsdl属w3c标准。
xml是webservice的跨平台的基础,XML主要的优点在于它既与平台无关,又与厂商无关。
XSD,W3C为webservice制定了一套传输数据类型,使用xml进行描述,即XSD(XML Schema Datatypes),任何编程语言写的webservice接口在发送数据时都要转换成webservice标准的XSD发送。
2.1.2 JAXM&SAAJ
JAXM(JAVA API For XML Message)主要定义了包含了发送和接收消息所需的API,SAAJ(SOAP With Attachment API For Java,JSR 67)是与JAXM 搭配使用的API,为构建SOAP 包和解析SOAP 包提供了重要的支持,支持附件传输等,JAXM&SAAJ 与JAX-WS 都是基于SOAP 的Web 服务,相比之下JAXM&SAAJ 暴漏了SOAP更多的底层细节,编码比较麻烦,而JAX-WS 更加抽象,隐藏了更多的细节,更加面向对象,实现起来你基本上不需要关心SOAP 的任何细节
2.1.3 JAX-RS
JAX-RS是JAVA 针对REST(Representation StateTransfer)风格制定的一套Web 服务规范,由于推出的较晚,该规范(JSR 311,目前JAX-RS 的版本为1.0)并未随JDK1.6 一起发行。
支持JAX-RS服务规范的框架有:
CXF——XFire和Celtix的合并(一个由IONA赞助的开源ESB,最初寄存在ObjectWeb上)。
Jersey——Sun公司的JAX-RS参考实现。
RESTEasy——JBoss的JAX-RS项目。
Restlet——也许是最早的REST框架了,它JAX-RS之前就有了。
注:REST 是一种软件架构模式,只是一种风格,rest服务采用HTTP 做传输协议
Webservice三要素
2.1.4 soap
SOAP即简单对象访问协议(Simple Object Access Protocal) 是一种简单的基于 XML 的协议,它使应用程序通过 HTTP 来交换信息,简单理解为soap=http+xml。
Soap协议版本主要使用soap1.1、soap1.2
2.1.5 wsdl
WSDL 是基于 XML 的用于描述Web Service及其函数、参数和返回值。通俗理解Wsdl是webservice的使用说明书。
2.1.6 UDDI
UDDI 是一种目录服务,通过它,企业可注册并搜索 Web services。企业将自己提供的Web Service注册在UDDI,也可以使用别的企业在UDDI注册的web service服务,从而达到资源共享。
UDDI旨在将全球的webservcie资源进行共享,促进全球经济合作。
但是使用webservice并不是必须使用UDDI,因为用户通过WSDL知道了web service的地址,可以直接通过WSDL调用webservice。
Webservice的应用场景
1. 企业中异构系统集成
在做企业整体信息化时,企业中一般都或多或少的存在一些既存系统,这些各种各样的系统不可能全部推翻,重新规划和开发,因为很多供应商在某一领域也做的很专业,博众家之长并进行集成应该是一个比较现实和可取的做法。各个系统之间通过WebService进行集成,不仅缩短了开发周期,降低了风险,还减少了代码复杂度,并能够增强应用程序的可维护性,因为webservice支持跨平台且遵循标准协议(soap)。
2. 软件扩展重用
将一个软件的功能以webservice方式暴露出来,达到软件重用。例如上边分析的天气预报,将天气查询功能以webservice接口方式暴露出来非常容易集成在其它系统中;再比如一个第三方物流系统将快递查询、快递登记暴露出来,从而集成在电子商务系统中。
3. 手机App服务端接口
现在几乎100%的手机App都需要联网和服务端有数据交互的行为。此时需要服务端开发服务接口也就是,Webservice接口。一般来说可以可使用rest风格的Webservice发布服务,手机App和服务之间传递xml或者json数据。
3 WSDL
Webservice的使用说明书。描述了webservice的服务地址以及webservice服务接口、参数、返回值。
Wsdl的阅读方法:从下往上读。
1、先找Service节点:每个wsdl中有且只有一个service节点,叫做服务视图节点。相当于插座的面板。Service节点中有port节点服务端端口。
2、根据port节点的binding属性找binding节点。根据binding节点的type属性找portType节点。
3、portType节点就是我们定义的SEI服务的接口类型。Prottype中的operation 节点就是方法名称。
4、operation 节点的input就是参数的定义,output就是返回值的定义。
5、Input有个属性叫做message,message属性对应message节点。其中有一个element,对应element节点。
6、Element节点定义中xsd中。定义了数据的类型。参数和返回值都在其中定义。
wsdl图示
schema
查询公网天气
3.1.1 实现步骤:
第一步:生成客户端调用代码
第二步:创建服务视图
第三步:从服务视图获得portType对象
第四步:调用服务端方法。
3.1.2 代码实现
public class WeatherClient2 {
public static void main(String[] args) {
//创建服务视图
WeatherWebService service = new WeatherWebService();
//从服务视图获得portType对象
WeatherWebServiceSoap portType = service.getWeatherWebServiceSoap();
//调用服务端方法查询天气
ArrayOfString arrayOfString = portType.getWeatherbyCityName("三亚");
//打印天气信息
for (String string : arrayOfString.getString()) {
System.out.println(string);
}
}
}
3.1.3 第二种调用webservice的方法
第一种方法的缺点是,如果服务器的ip或者域名发生改变后需要重新生成客户端调用代码。需要把ip或者的域名配置到配置文件中,至少是可修改的。
使用Service类创建服务视图
public class WeatherClient {
public static void main(String[] args) {
//创建服务视图
//指定服务的url,可以是wsdl的地址也可以是服务的地址
URL url = null;
try {
url = new URL("http://www.webxml.com.cn/WebServices/WeatherWebService.asmx?WSDL");
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//第一个参数是wsdl的命名空间
//第二个参数是service节点的name属性
QName qName = new QName("http://WebXml.com.cn/", "WeatherWebService");
Service service = Service.create(url, qName);
//从服务视图获得portType对象
//WeatherWebServiceSoap portType = service.getWeatherWebServiceSoap();
WeatherWebServiceSoap portType = service.getPort(WeatherWebServiceSoap.class);
//调用服务端方法查询天气
ArrayOfString arrayOfString = portType.getWeatherbyCityName("三亚");
//打印天气信息
for (String string : arrayOfString.getString()) {
System.out.println(string);
}
}
}
4 Soap
soap是什么
SOAP 是一种网络通信协议
SOAP即Simple Object Access Protocol简易对象访问协议
SOAP 用于跨平台应用程序之间的通信
SOAP 被设计用来通过因特网(http)进行通信
SOAP = HTTP+XML,其实就是通过HTTP发xml数据
SOAP 很简单并可扩展支持面向对象
SOAP 允许您跨越防火墙
SOAP 将被作为 W3C 标准来发展
使用TCP/IP Monitor监视Soap协议
TCP/IPMonitor是eclipse自带的工具。
是一个代理方式的查看协议体内容的工具。
Soap1.1
Webservice默认使用的是1.1版本的soap协议。
4.1.1 请求的协议体
头信息
POST /weather HTTP/1.1
Accept: text/xml, multipart/related
Content-Type: text/xml; charset=utf-8
SOAPAction: "http://jaxws.itheima.com/WeatherInterfaceImpl/queryWeatherRequest"
User-Agent: JAX-WS RI 2.2.4-b01
Host: 127.0.0.1:54321
Connection: keep-alive
Content-Length: 210
<?xml version="1.0" ?>
<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
<S:Body>
<ns2:queryWeather xmlns:ns2="http://jaxws.itheima.com/">
<arg0>北京</arg0>
</ns2:queryWeather>
</S:Body>
</S:Envelope>
4.1.2 响应的协议体
HTTP/1.1 200 OK
Transfer-encoding: chunked
Content-type: text/xml; charset=utf-8
Date: Wed, 24 Jun 2015 07:21:20 GMT
<?xml version="1.0" ?>
<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
<S:Body>
<ns2:queryWeatherResponse xmlns:ns2="http://jaxws.itheima.com/">
<return>雷阵雨</return>
</ns2:queryWeatherResponse>
</S:Body>
</S:Envelope>
结论就是:soap=http+xml
Soap1.2协议
如果使用soap1.2协议的话只需要在SEI实现类上添加一个注解即可。
@BindingType(javax.xml.ws.soap.SOAPBinding.SOAP12HTTP_BINDING)
Jdk自带的jax-ws API(JAX-WS RI 2.2.4-b01)不支持soap1.2版本的协议需要使用高版本的api。
需要使用到2.2.8版本的jax-ws的api。
使用soap1.2后wsdl的变化:
soap变成soap12
Soap协议协议改变后,不需要重新生成客户端调用代码。
4.1.3 请求的协议体
POST /weather HTTP/1.1
Accept: application/soap+xml, multipart/related
Content-Type: application/soap+xml; charset=utf-8;
action="http://jaxws.itheima.com/WeatherInterfaceImpl/queryWeatherRequest"
User-Agent: JAX-WS RI 2.2.4-b01
Host: 127.0.0.1:54321
Connection: keep-alive
Content-Length: 208
<?xml version="1.0" ?>
<S:Envelope xmlns:S="http://www.w3.org/2003/05/soap-envelope">
<S:Body>
<ns2:queryWeather xmlns:ns2="http://jaxws.itheima.com/">
<arg0>北京</arg0>
</ns2:queryWeather>
</S:Body>
</S:Envelope>
4.1.4 响应的协议体
HTTP/1.1 200 OK
Transfer-encoding: chunked
Content-type: application/soap+xml; charset=utf-8
Date: Wed, 24 Jun 2015 07:59:16 GMT
<?xml version='1.0' encoding='UTF-8'?>
<S:Envelope xmlns:S="http://www.w3.org/2003/05/soap-envelope">
<S:Body>
<ns2:queryWeatherResponse xmlns:ns2="http://jaxws.itheima.com/">
<return>雷阵雨</return>
</ns2:queryWeatherResponse>
</S:Body>
</S:Envelope>
Soap1.1和soap1.2的区别
Content-type:
Soap1.1:Content-type: text/xml; charset=utf-8
Soap1.2:Content-type: application/soap+xml; charset=utf-8
Envelope标签的命名空间:
Soap1.1:<S:Envelopexmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
Soap1.2:<S:Envelopexmlns:S="http://www.w3.org/2003/05/soap-envelope">
模拟httpclient调用webservice
4.1.5 实现步骤
第一步:创建一java工程
第二步:创建一个HttpURLConnection对象。可以使用URL对象的openConnection方法获得。
第三步:设置content-type,根据服务端使用的版本,设置不同的content-type。
第四步:编辑一个请求的xml格式的协议体发送给服务端。
第五步:接收服务端响应的内容。打印出来
4.1.6 代码的实现
使用http的方式调用服务端方法
public class SoapHttpClient {
public static void main(String[] args) throws Exception {
URL url = new URL("http://127.0.0.1:54321/weather");
//创建一个HttpURLConnection对象
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
//设置content-type
//服务端是soap1.2的协议
connection.setRequestProperty("Content-Type", " application/soap+xml; charset=utf-8;");
//设置使用connection对象进行输入输出
connection.setDoInput(true);
connection.setDoOutput(true);
//使用输出流向服务端发送xml数据
connection.getOutputStream().write(getRequestBody("北京").getBytes());
//接收服务端响应的内容
InputStream inputStream = connection.getInputStream();
byte[] b = new byte[1024];
int len = 0;
ByteArrayOutputStream data = new ByteArrayOutputStream();
while((len = inputStream.read(b, 0, 1024)) != -1) {
data.write(b, 0, len);
}
System.out.println(data);
}
//拼装请求协议体
private static String getRequestBody(String cityName) {
String body = "<?xml version=\"1.0\" ?>\n" +
"<S:Envelope xmlns:S=\"http://www.w3.org/2003/05/soap-envelope\">\n" +
" <S:Body>\n" +
" <ns2:queryWeather xmlns:ns2=\"http://jaxws.itheima.com/\">\n" +
" <arg0>"+cityName+"</arg0>\n" +
" </ns2:queryWeather>\n" +
" </S:Body>\n" +
"</S:Envelope>";
return body;
}
}
拼装xml字符串小技巧
5 案例:区域查询系统
5.1区域查询系统的结构
6.2需求
创建区域查询服务系统,对外发布WebService服务,供客户端调用,根据parentid查询区域信息。客户端向服务端传递xml格式数据,服务端向客户端响应xml格式数据。
6.3分析
基于soap协议的webservice就可以传递对象。
5.1.1 传递xml数据的原因:
1、跨语言时可能会花很长时间来调试,如果直接传递xml省去的调试的时间。参数和返回值都是字符串类型,非常简单。
2、如果参数发生变化后,可以不用修改接口,不用重新生成客户端调用代码。
3、Xml格式的数据是跨语音跨平台的。
5.1.2 架构分析
5.1.3 实现步骤:
服务端:
第一步:创建一个java工程
第二步:导入jar包需要mysql的数据库驱动,dom4j的jar包。
第三步:创建一个SEI。
第四步:创建SEI实现类。调用dao查询区域列表。
1、接收客户端发送的xml
2、需要解析xml数据转换成java对象
3、查询数据库得到区域列表,使用jdbc查询。
4、把区域列表转换成xml数据
5、返回xml数据
第五步:发布服务,使用Endpoint的publish方法发布服务。
客户端:
1、生成客户端调用代码
2、创建服务视图
3、从服务视图获得porttype
4、调用服务端方法。
5.1.4 接口描述:
客户端发送数据格式:
<?xml version="1.1" encoding="utf-8"?>
<queryarea>
<parentid> </parentid>//父级区域id
<start></start>//起始记录,从1开始
<end></end>//结束记录
</queryarea>
服务端响应数据格式:
<?xml version="1.0" encoding="UTF-8"?>
<areas>
<area>
<areaid> </areaid>//区域id
<areaname></areaname>//区域名称
<arealevel></arealevel>//区域等级
<parentid></parentid>//父级区域id
</area>
//…..
</areas>
5.1.5 代码实现
5.1.5.1 Dao
public class AreaDao {
public List<AreaModel> queryArea(String parentid, int start, int end) {
Connection connection = null;
PreparedStatement preparedStatement = null;
ResultSet resultSet = null;
List<AreaModel> areaList = new ArrayList<>();
try {
//加载数据库驱动
Class.forName("com.mysql.jdbc.Driver");
//获得connection
connection = DriverManager.getConnection("jdbc:mysql:///webservice", "root", "root");
//获得preparedStatement
String sql = "select * from area where parentid = ? limit ?,?";
preparedStatement = connection.prepareStatement(sql);
//设置参数
preparedStatement.setString(1, parentid);
preparedStatement.setInt(2, start - 1);
preparedStatement.setInt(3, end - start + 1);
//执行查询
resultSet = preparedStatement.executeQuery();
//取区域列表
while(resultSet.next()) {
AreaModel model = new AreaModel();
model.setAreaid(resultSet.getString("areaid"));
model.setAreaname(resultSet.getString("areaname"));
model.setArealevel(resultSet.getString("arealevel"));
model.setParentid(resultSet.getString("parentid"));
//添加到区域列表
areaList.add(model);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
resultSet.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
try {
preparedStatement.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
try {
connection.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return areaList;
}
}
5.1.5.2 Service
@WebService
public class AreaInterfaceImpl implements AreaInterface {
@Override
public String queryArea(String area) {
//解析xml查询条件
AreaModel model = null;
String result = null;
try {
model = parseXml(area);
AreaDao dao = new AreaDao();
List<AreaModel> list = dao.queryArea(model.getParentid(), model.getStart(), model.getEnd());
result = list2xml(list);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return result;
}
/*
<?xml version="1.1" encoding="utf-8"?>
<queryarea>
<parentid> </parentid>//父级区域id
<start></start>//起始记录,从1开始
<end></end>//结束记录
</queryarea>
*/
private AreaModel parseXml(String xml) throws Exception {
Document document = DocumentHelper.parseText(xml);
String parentid = document.selectSingleNode("/queryarea/parentid").getText();
String start = document.selectSingleNode("/queryarea/start").getText();
String end = document.selectSingleNode("/queryarea/end").getText();
AreaModel model = new AreaModel();
model.setParentid(parentid);
model.setStart(Integer.parseInt(start));
model.setEnd(Integer.parseInt(end));
return model;
}
/**
<?xml version="1.0" encoding="UTF-8"?>
<areas>
<area>
<areaid> </areaid>//区域id
<areaname></areaname>//区域名称
<arealevel></arealevel>//区域等级
<parentid></parentid>//父级区域id
</area>
//…..
</areas>
*/
private String list2xml(List<AreaModel> list) throws Exception {
Document document = DocumentHelper.createDocument();
//添加以根节点
Element root = document.addElement("areas");
for (AreaModel areaModel : list) {
Element area = root.addElement("area");
area.addElement("areaid").setText(areaModel.getAreaid());
area.addElement("areaname").setText(areaModel.getAreaname());
area.addElement("arealevel").setText(areaModel.getArealevel());
area.addElement("parentid").setText(areaModel.getParentid());
}
return document.asXML();
}
}
5.1.5.3 发布服务
public class AreaServer {
public static void main(String[] args) {
Endpoint.publish("http://127.0.0.1:12345/area", new AreaInterfaceImpl());
}
}
5.1.5.5 生成客户端调用代码
5.1.5.6 客户端
public class AreaClient {
public static void main(String[] args) {
//创建服务视图
AreaInterfaceImplService service = new AreaInterfaceImplService();
//获得porttype
AreaInterfaceImpl portType = service.getAreaInterfaceImplPort();
//调用服务端方法
String result = portType.queryArea(getQueryXml("1.2.", 1, 10));
System.out.println(result);
}
private static String getQueryXml(String parentid, int start, int end) {
String xml = "<?xml version=\"1.1\" encoding=\"utf-8\"?>\n" +
"<queryarea>\n" +
"<parentid>"+parentid+"</parentid>\n" +
"<start>"+start+"</start>\n" +
"<end>"+end+"</end>\n" +
"</queryarea>";
return xml;
}
}