因項目需要,將之前使用過的webservice重新撿了起來,並且這次選擇了使用不需要jar包的Java原生Jaxws。首先是wsimport的用法,先分享我的用法:cmd下先轉到工程所在路徑,然後運行以下命令
代碼如下:
wsimport -keep -extension -s ./src -p com.jaxws.test http://192.168.1.1:8080/service?wsdl
在com.jaxws.test的包下就能找到自動生成的各種客戶端相關輔助類。這些類怎麼用就不細說了,網上很多資料。然後就是自己編寫調用類(我這裡只是一個方法,會被調用到就行)
代碼如下:
public String jaxws(Object[] opArgs)
{
ServicesService service=new ServicesService();
//向SOAP添加表頭
service.setHandlerResolver(new HandlerResolver(){
public List<Handler> getHandlerChain(PortInfo portInfo) {
List<Handler> handlerList = new ArrayList<Handler>();
//添加認證信息
handlerList.add(new ClientHandler());
return handlerList;
}
});
String result =service.getServicesPort().getResults(opArgs.toString());
//得到結果
System.out.println(result);
return result;
}
注意到這裡的【//添加認證信息 handlerList.add(new ClientHandler());】,所以我們還需要新建一個ClientHandler類來實現認證消息的組裝,如下:
代碼如下:
package com.jaxws.test;
import java.util.Set;
import javax.xml.namespace.QName;
import javax.xml.soap.*;
import javax.xml.ws.handler.MessageContext;
import javax.xml.ws.handler.soap.SOAPHandler;
import javax.xml.ws.handler.soap.SOAPMessageContext;
public class ClientHandler implements SOAPHandler<SOAPMessageContext> {
public boolean handleMessage(SOAPMessageContext ctx) {
//出站,即客戶端發出請求前,添加表頭信息
Boolean request_p=(Boolean)ctx.get(MessageContext.MESSAGE_OUTBOUND_PROPERTY);
if(request_p){
try {
SOAPMessage msg=ctx.getMessage();
SOAPEnvelope env=msg.getSOAPPart().getEnvelope();
SOAPHeader hdr=env.getHeader();
if(hdr==null) hdr=env.addHeader();
//添加認證信息頭
//QName(String namespaceURI, String localPart, String prefix)
//QName(String namespaceURI, String localPart)
//QName(String localPart)
//@param namespaceURI:QName的名稱空間
//@param localPart:QName的本地部分
//@param prefix:QName的前綴
QName name=new QName("http://csdc.info/", "Authentication", "wsse");
SOAPHeaderElement header = hdr.addHeaderElement(name);
//addChildElement(String localName, String prefix,String uri)
//addChildElement(String localName, String prefix)
//addChildElement(String localName)
//@param uri:新元素所屬空間名稱URI
//@param localName:新元素的本地名稱
//@param prefix:新元素名稱的空間前綴
//見JDK 1.6的API
SOAPElement userElement = header.addChildElement("Username", "wsse");
userElement.addTextNode("admin");
SOAPElement passElement = header.addChildElement("Password", "wsse");
passElement.addTextNode("admin");
msg.saveChanges();
//把SOAP消息輸出到System.out,即控制台
msg.writeTo(System.out);
return true;
} catch (Exception e) {
e.printStackTrace();
}
}
return false;
}
@Override
public boolean handleFault(SOAPMessageContext context) {
// TODO Auto-generated method stub
return false;
}
@Override
public void close(MessageContext context) {
// TODO Auto-generated method stub
}
@Override
public Set<QName> getHeaders() {
// TODO Auto-generated method stub
return null;
}
}
這個類就把所有的soap消息都加上了一個頭消息,我這裡的頭消息如下:
代碼如下:
<wsse:Authentication xmlns:wsse="http://csdc.info/">
<wsse:Username>admin</wsse:Username>
<wsse:Password>admin</wsse:Password>
</wsse:Authentication>
這樣就實現了帶soap頭認證的基於jaxws的webservice客戶端。