java实现webservice(java实现webserver)
如何在Java中快速发布WebService服务
在Java中快速发布WebService服务方法:
一、通过Axis2提供的模板自动发布
这种方法非常简单,只要下载Axis包后从里面的“axis-1_4\webapps”中找到axis站点,将它拷出来放到你要部署的tomcat/webapps下,再将要发布的java类放到axis文件夹里,再将文件名的后缀改为“jws”,这样一个webservice就发布成功了。访问时按照路径“类名.jws?wsdl”就可以了。
怎么样?这种方法非常是不是非常简单?对,它特别适合初学java或者是手头没有很好的SDE开发和调试工具的时候。
但很明显,这种方法不太好调试,又不能凸显您的专业水准,更不能集成到您的开发环境里。所以如果您不是很急,就可以用下面一种办法来发布:
二、通过JWS来手动发布
步骤如下:
1、在Eclipse里新建一个Dynamic Web Project,如下图所示(相信这一步您比我还熟,所以会的可以略过):
2、在“Project Name”中输入工程名称,如“JavaWebService”:
3、一直单击“Next”按钮直到出现如下对话框:
在上面的对话框中一定要注意需要选中“Generate web.xml deployment descriptor”,否则后面再添加就比较麻烦。选中后单击“Finish”按钮即可完成工程创建。
4、引入必须的jar包:
axis.jar、commons-discovery-0.2.jar、commons-logging.jar、jaxrpc.jar、saaj.jar、wsdl4j.jar
最好将jar包拷贝到工程的WEB-INF/lib下后直接引用本工程内的jar包,这样不会导致工程文件夹或jar包所在文件夹移动后读取不到jar包的问题:
分别单击顶层对话框和“Properties”对话框的“OK”按钮完成jar包引用。
5、新建package和java文件,如下图所示:
首先说说com.test.javabean.Student.java这个类。看得出来这个类是个JavaBean,目的是为了封装一个Java的实体类,这样做的好处是万一要把很多变量返回给客户端的话,客户端不至于在方法中声明很多形参来接收,而是通过一个对象来接收。Student.java的实现如下:
[java] view plain copy
package com.test.javabean;
import java.io.Serializable;
public class Student implements Serializable{
private String ID;
private String Name;
private String Sex;
private int age;
public String getID() {
return ID;
}
public void setID(String iD) {
ID = iD;
}
public String getName() {
return Name;
}
public void setName(String name) {
Name = name;
}
public String getSex() {
return Sex;
}
public void setSex(String sex) {
Sex = sex;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
再来看看com.test.webservice.CStudent.java这个类,定义如下:
[java] view plain copy
package com.test.webservice;
import javax.jws.WebService;
import com.test.javabean.Student;
@WebService
public class CStudent {
public Student getStudent(String name) throws java.rmi.RemoteException{
Student student = new Student();
student.setName(name);
student.setAge(14);
student.setID("25");
student.setSex("Male");
return student;
}
}
为了简单起见,就不涉及到更多的后台查询了,先写死返回结果吧。不知您注意到了没有,这个类跟普通类的区别是在类的顶部加了个“@WebService”声明。对,这样就是个WebService方法了,是不是也很简单?但是这样就得记着在上面引入javax.jws.WebService包。偶对了,这种方式是JDK1.7里增加的方式,所以要想使用这种特性,就得麻烦您把JDK升级到1.7,方法很简单,用绿色版的就可以啦,不用劳烦您把JDK卸了又重装。只需要把JDK放到哪个盘的根目录(这样是为了防止路径中有空格或是括号什么的,否则一旦出问题您就苦逼了),然后把JDK的路径定义为JAVA_HOME,再在path中增加%JAVA_HOME%/bin;就可以了。这样做的好处是可以随时很方便地更换JDK的版本,而不需要卸载后再重新安装。
扯远了。有了WebService的方法体,您还需要再发布一下WebService,所以还得新建一个com.test.servlet.CStudentServlet.java类:
[java] view plain copy
package com.test.servlet;
import java.io.IOException;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Properties;
import javax.jws.WebService;
import javax.servlet.*;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.xml.ws.Endpoint;
import com.test.webservice.CStudent;
public class CStudentServlet extends HttpServlet{
/**
* @see Servlet#init(ServletConfig)
*/
public void init(ServletConfig config) throws ServletException {
// TODO Auto-generated method stub
super.init(config);
System.out.println("正在启动WebService服务:");
Endpoint.publish(";span style="font-family: Arial, Helvetica, sans-serif;"192.168.56.1/spanspan style="font-family: Arial, Helvetica, sans-serif;":8082//spanspan style="font-family: Arial, Helvetica, sans-serif;"JavaWebService/spanspan style="font-family: Arial, Helvetica, sans-serif;"/Student", new CStudent());/span
}
/**
* @see Servlet#getServletConfig()
*/
public ServletConfig getServletConfig() {
// TODO Auto-generated method stub
return null;
}
/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
}
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("进入WSServlet的service()方法");
}
/**
* @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
*/
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
}
}
上面的代码中有处细节不知您注意了没有:把IP地址、端口号、站点名等信息写死到了代码里。对,不仅如此,你这里写的什么IP,就必须用什么IP访问,即便是在本机写成localhost都不可以。jws就是这么奇怪。所以在实际中可以参考一些资料动态获取IP地址和端口号。
对了,您那边代码是不是报错了?对,这是因为您还得做一步操作——添加server和对server组件包的引用,以tomcat为例,方法如下:
找到servers窗口,单击“new server wizard...”链接:
打开如下对话框:
看到了吧?选择Apache下的”Tomcat v7.0 Server“,然后单击”Next“:
在上图中单击右侧”Browse“按钮,找到Tomcat所在的目录,再在JRE中选择1.7的JDK。如果您还没有配置JDK,就请自行查找资料进行配置吧,某在这里就不讲了,不然又扯远啦。
然后进入这个窗口:
这个对话框得特别留意一下,不要着急单击”Add“按钮添加站点那,否则启动后您就找不到站点部署的位置了。所以这一步直接单击”Finish“。完毕后双击新添加的server,打开如下对话框:
在上面的窗口中,一定要先选择第二项”Use Tomcat installation (takes control of Tomcat installation)“,然后在”Deploy path“中将内容改为”webapps“,然后再按下”Ctrl+S“键保存配置,这样就可以把站点部署到您指定的tomcat的webapps下了。
然后记着给站点添加servlet-api.jar包的引用,在工程上单击右键,选择”Properties“,打开如下对话框:
看到了吧?切换到”Libraries“选项卡后先点击右侧的”Add Library“按钮,再在弹出的窗口中选择”Server Runtime“,单击”Next“--“Finish”和父窗口的”OK“按钮就可以成功引入servlet所需的jar包了。怎么样?CStudentServlet.java中再没有错误提示了吧?
下一步需要将servlet配置为自启动,您只要在web.xml中增加如下内容就可以了:
[html] view plain copy
servlet
servlet-nameCStudentServlet/servlet-name
servlet-classcom.test.servlet.CStudentServlet/servlet-class
load-on-startup1/load-on-startup
/servlet
servlet-mapping
servlet-nameCStudentServlet/servlet-name
url-pattern/servlet/CStudentServlet/url-pattern
/servlet-mapping
记着把站点发布一下,在tomcat上单击邮件,选择“Add and Remove”,将JavaWebService发布到tomcat中。
下一步只要发布和启动一下站点就可以了。如果您还不会,那就在server上单击右键,选择“Publish”,然后再选择“Start”就可以了。
启动成功
forest调用webservice
用java实现web services必须遵循两个规范,他们分别是:
? JSR 101:用于基于XML的RPC ,Remote Procedure Call、 的Java API 、Java
API for XML based RPC, JAX-RPC,
? JSR 109:实现企业Web服务。Implementing Enterprise Web services。两个规范提供了厂商的实现的一致性和互操作性需求。
JAX-RPC——Java到XML和XML到Java映射API
JAX-RPC为基于XML的远程过程调用。Remote Procedure Call、 RPC,和Java应用程序编程接口 、Java Application Programming Interface。 API, 、
? WSDL到Java和Java到WSDL映射,例如、将WSDL端口类型映射到Java服
务端点接口 ,Java Service Endpoint Interface、 SEI。 。
? XML数据类型到Java数据类型和Java数据类型到XML数据类型映射。包括简
单类型、复杂类型和数组。
除了XML映射之外, JAX-RPC还定义了服务器端编程模型和API。 AX-RPC 1. 1根据Web 服务互操作性组织、Web Services Interoperability organization、 WS-I。基本概要版本1.0 、Basic Profile version 1.0,添加了互操作性需求。
JSR 109——J2EE环境的API
JSR 109指定了Java 2 Enterprise Edition ,J2EE、环境的Web服务编程模型和体系结构。 JSR 109构建在SOAP 1. 1和WSDL 1. 1的基础上、它涵盖了J2EE环境中JAX-RPC 的使用, 图2, 。它还定义了J2EE应用程序服务器中的部署模型。 JSR 109的客户端编程模型符合JAX-RPC。
JAX-RPC 1. 1和JSR 109是J2EE 1.4的组成部分。
1/7页
在这里我不准备描述用java如何实现服务端,仅描述如何用java作为客户端如何调用web services。
获取客户端服务
Java客户端有两种获取方式、一种是基于容器管理的、一种不是基于容器管理的。基于容器管理的、可以是不同类型的J2EE容器管理的客户机,
? 应用程序客户机容器客户机
? Web容器客户机。 JavaBean或Servlet
? EJB容器客户机、 EJB
他们通过jndi查找服务。一般是两个步骤:
1,实例化本地JND I上下文。
2、在此上下文中对Web服务进行JND I查找。
Context ic = new InitialContext() ;
Service service = (Service) ctx. lookup("java:comp/env/service/HelloService") ;然后可以使用获得的服务接口 。Service Interface,来获取静态存根、动态代理或DII Call对象,来实现我下面描述的三种调用方式。
不是基于容器管理的、就是在本地通过java程序直接调用服务的,服务查找是通过JAX-RPC ServiceFactory进行的、 JAX-RPC ServiceFactory是创建服务访问点的工厂,可以如下进行使用获取服务。javax.xml.rpc.Service service =
ServiceFactory.newInstance() .createService(. . . ) ;
我在下面描述的客户端获取方式都是基于不是容器管理的。java调用webservices
用java作为客户端调用webservices主要有三种方式,下面描述三种调用方式。
1.DLL 、Dynamic Invocation Interface,调用
复杂度最高。灵活性最强的调用方式
通过自己创建Call对象完成调用,此方法的步骤如下,
2/7页
1. 获取一个JAX-RPC Service 。
2. 使用JAX-RPC Service的createCall ()方法实例化JAX-RPC Call 。
3. 使用它的setter方法来配置您的Call实例。
4. 使用JAX-RPC Call的调用方法来调用Web服务的操作。
String namespace = "";
String portName = "Hello";
QName portQN = new QName"(namespace, "portName) ;
String operationName = getGreeting ;
Call call = service.createCall () ;call. setPortTypeName(portQN) ;call. setOperationName(new QName(namespace, operatio"n"Name) ) ;call. setProperty(Call.ENCODINGSTYLE_URI_PROPERTY, ) ;call. setProperty(Ca"ll.OPER"ATION_STYLE_PROPERTY, "wrapped") ;call.addParameter( param1 , xsd:string,ParameterMode. IN) ;call. setReturnType(xsd:string) ;
Object[] inParams = new Object[] {"Jane"} ;
String ret = (String) call. invoke(inParams) ;
使用DII Call接口的优势在于,客户机可以调用远程过程而无需知道开发时的WSDL URI或Web服务操作的签名。这样当Web服务的细节改变时、很容易对代码进行修改。使用DII客户机,不需要像动态代理或静态存根的情形那样由从WSDL到Java的映射工具。Emitter,生成运行时类。然而,如果您知道您想要调用的 Web服务不可能更改,就应该使用动态代理。 因为配置Call实例可能很复杂。
我以调用一个axis的webservices为例说明DLL如何调用。
编写服务端程序Hello package com. lion. service public class Hello {public String sayHello(String name)
{return "hello "+name;
}
}
将上面的程序发布成一个web服务。在web-info\server-config.wsdd文件添加下列描述,service name="Hel"lo" provide"r="java":RPC"
parameter name= className value= com. lion. service .Hello"/
parameter name="allowedMethods" value=" sayHello "/
/service
客户端调用程序如下TestHelloClient. java,import org.apache.axis.client.Call ;
3/7页
import org.apache.axis.client.Service;import javax.xml.namespace.QName;import javax.xml.rpc.ServiceException;import java.net.MalformedURLException;import java.rmi.RemoteException;public class TestHelloClient{public static void main(String[] args) {try {
String endpoint = " ";
Service service = new Service() ;
Call call = null;call = (Call) service.createCall () ;call. setTargetEndpointAddress(new java.net.URL(endpoint) ) ;
//为Call设置服务的位置call. setOperationName( “sayHello” ) ;
String res = (String)" call. invoke(new"Object[] {"lion"} ) ;
System.out.println( return value is + res) ;
} catch (Exception ex) {ex.printStackTrace() ;
}
}
}
2.使用动态代理,Dynamic Proxy、调用
复杂度中等、灵活性中等的调用方式
您可以使用代理从JAX-RPC Service中调用Web服务的操作。代理是实现SEI (服务端点接口 Service Endpoint Interface)的Java类。获得代理使用的是JAX-RPC Service 的getPort()方法,它接受您想要调用的Web服务的端口的名称,存在于WSDL文档中。以及代理实现的SEI。它之所以称为动态是因为该代理是在运行时创建的。动态代理客户机的步骤如下、
1. 获取一个JAX-RPC Service 。
2. 使用JAX-RPC Service的getPort()方法来获得一个代理以调用Web服务的操
作。
String namespace = "";
String portName = "Hello";
QName portQN = new QName(namespace, portName) ;
4/7页
Hello myProxy = service.getPort(portQN, Hello.class) ;
System.out.println(myProxy.getGreeting("Jane") ) ;
这是所有您为了使用动态代理方法调用Web服务而需要编写的代码。使用这种方法的优势在于您可以编写可移植的、厂商无关的代码。然而、您需要知道开发时的WSDL URL、并且需要在运行之前根据WSDL文档运行您的从WSDL到Java的映射工具。如果您没有这方面的信息,或者WSDL URL很可能改变、那么您应该改为使用DII方法。
仍以上面发布的hello服务为例子。下面是动态代理的调用代码。
编写客户端代理接口
//这里声明的方法一定要和服务端程序的方法名称和参数相同。最好类名也一样//而且必须继承java.rmi.Remote接口public interface HelloClientInterface extends java.rmi.Remote {public String sayHello (String name) throws java.rmi.RemoteException;}
编写并执行客户端程序TestHelloClient. java import javax.xml.rpc.Service;import javax.xml.rpc.ServiceFactory;import java.net.URL;import javax.xml.namespace.QName;public class TestHelloClient {public static void main(String[] args) {try
{
String wsdlUrl = " ";//wsdl描述
String nameSpaceUri = " ";//wsdl描述里面的服务命名空间
String serviceName = "HelloService";// wsdl描述里面发布的服务名称
String portName = "Hello";//wsdl描述里面面发布的服务里面的port名称
ServiceFactory serviceFactory = ServiceFactory.newInstance() ;
Service afService = serviceFactory.createService(new URL(wsdlUrl) ,new QName(nameSpaceUri, serviceName) ) ;
HelloClientInterface proxy = (HelloClientInterface)afService.getPort(new QName(nameSpaceUr"i, portName) , Hel"loClientInterface."clas"s) ;
System.out.println( return value is +proxy. sayHello( lion ) ) ;
}catch(Exception ex)
{ex.printStackTrace() ;
}
java如何调用webservice接口
Java调用WebService可以直接使用Apache提供的axis.jar自己编写代码,或者利用Eclipse自动生成WebService Client代码,利用其中的Proxy类进行调用。理论上是一样的,只不过用Eclipse自动生成代码省事些。
1、编写代码方式:
package com.yudun.test;
import java.rmi.RemoteException;
import org.apache.axis.client.Call;
import org.apache.axis.client.Service;
import org.apache.axis.message.PrefixedQName;
import org.apache.axis.message.SOAPHeaderElement;
import com.cezanne.golden.user.Exception;
import com.cezanne.golden.user.UserManagerServiceProxy;
import javax.xml.namespace.QName;
import java.net.MalformedURLException;
import javax.xml.rpc.ServiceException;
import javax.xml.soap.Name;
import javax.xml.soap.SOAPException;
public class testWebService {
public static String getResult() throws ServiceException, MalformedURLException, RemoteException, SOAPException
{
//标识Web Service的具体路径
String endpoint = "WebService服务地址";
// 创建 Service实例
Service service = new Service();
// 通过Service实例创建Call的实例
Call call = (Call) service.createCall();
//将Web Service的服务路径加入到call实例之中.
call.setTargetEndpointAddress( new java.net.URL(endpoint) );//为Call设置服务的位置
// 由于需要认证,故需要设置调用的SOAP头信息。
Name headerName = new PrefixedQName( new QName("发布的wsdl里的targetNamespace里的url", "string_itemName") );
org.apache.axis.message.SOAPHeaderElement header = new SOAPHeaderElement(headerName);
header.addTextNode( "blablabla" );
call.addHeader(header);
// SOAPHeaderElement soapHeaderElement = new SOAPHeaderElement("发布的wsdl里的targetNamespace里的url", "SoapHeader");
// soapHeaderElement.setNamespaceURI("发布的wsdl里的targetNamespace里的url");
// try
// {
// soapHeaderElement.addChildElement("string_itemName").setValue("blablabla");
// }
// catch (SOAPException e)
// {
// e.printStackTrace();
// }
// call.addHeader(soapHeaderElement);
//调用Web Service的方法
org.apache.axis.description.OperationDesc oper;
org.apache.axis.description.ParameterDesc param;
oper = new org.apache.axis.description.OperationDesc();
oper.setName("opName");
param = new org.apache.axis.description.ParameterDesc(new javax.xml.namespace.QName("", "arg0"), org.apache.axis.description.ParameterDesc.IN, new javax.xml.namespace.QName("", "string"), java.lang.String.class, false, false);
param.setOmittable(true);
oper.addParameter(param);
param = new org.apache.axis.description.ParameterDesc(new javax.xml.namespace.QName("", "arg1"), org.apache.axis.description.ParameterDesc.IN, new javax.xml.namespace.QName("", "string"), java.lang.String.class, false, false);
param.setOmittable(true);
oper.addParameter(param);
param = new org.apache.axis.description.ParameterDesc(new javax.xml.namespace.QName("", "arg2"), org.apache.axis.description.ParameterDesc.IN, new javax.xml.namespace.QName("", "string"), java.lang.String.class, false, false);
param.setOmittable(true);
oper.addParameter(param);
oper.setReturnType(new javax.xml.namespace.QName("", "string"));
oper.setReturnClass(java.lang.String.class);
oper.setReturnQName(new javax.xml.namespace.QName("", "return"));
oper.setStyle(org.apache.axis.constants.Style.WRAPPED);
oper.setUse(org.apache.axis.constants.Use.LITERAL);
oper.addFault(new org.apache.axis.description.FaultDesc(
new javax.xml.namespace.QName("发布的wsdl里的targetNamespace里的url", "Exception"),
"Exception",
new javax.xml.namespace.QName("发布的wsdl里的targetNamespace里的url", "Exception"),
true
));
call.setOperation( oper );
call.setOperationName(new javax.xml.namespace.QName("发布的wsdl里的targetNamespace里的url", "opName"));
//调用Web Service,传入参数
String res = ( String ) call.invoke( new Object[]("arg0","arg1"));
System.out.println("===============");
return res;
}
/**
* @param args
*/
public static void main(String[] args) {
try {
System.out.println(getResult());
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (RemoteException e) {
e.printStackTrace();
} catch (ServiceException e) {
e.printStackTrace();
} catch (SOAPException e) {
e.printStackTrace();
}
}
}
2、利用Eclipse自动生成WebService client代码就容易多了:(由于还不会发图片,就用语言描述了,大家酬和看吧。。。)
首先,new project,选择other,在输入框中输入Web Service Client,选中搜索后的结果,点击Next,在Service definition中输入 WebService的发布地址,点击Finish
这样,WebService Client代码已经生成好了。
接下来写一个Test类,在main函数中输入如下代码:
String endpoint = "服务器的WebService的地址";
YourWebServiceNameProxy umsp = new YourWebServiceNameProxy (endpoint);
try {
String resultStr = umsp.opMethod("arg0","arg1");
System.out.println(resultStr);
} catch (Exception e) {
System.out.println("异常");
e.printStackTrace();
} catch (RemoteException e) {
System.out.println("RemoteException异常");
e.printStackTrace();
}
如果还有疑问的话还有视频,如果对你有帮助请采纳!
