欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

使用CXF和Jersey框架来进行Java的WebService编程

程序员文章站 2024-03-08 17:56:52
cxf cxf是在xfire的基础上实现的。 1)首先呢,还是包的问题,在http://cxf.apache.org/download.html这里可以下到最新版的cx...

cxf
cxf是在xfire的基础上实现的。
1)首先呢,还是包的问题,在http://cxf.apache.org/download.html这里可以下到最新版的cxf,当然,我用的是最新版的。接下来还是那句废话,建web项目,放入jar包。而jar包我们就不选择了,一堆全部放入。
我们会看到它包含了spring的jar包,后面当我们需要把cxf作为web项目部署时,就需要用到spring的配置文件,这个后面再讲。
还是接口类和实现类:

@webservice 
public interface ireaderservice { 
  public reader getreader(@webparam(name="name") string name,@webparam(name="password") string password); 
  public list<reader> getreaders(); 
} 
@webservice(endpointinterface="com.cxf.servlet.ireaderservice",servicename="readerservice") 
public class readerservice implements ireaderservice{ 
  public reader getreader(@webparam(name="name") string name,@webparam(name="password") string password) { 
    return new reader(name,password); 
  } 
   
  public list<reader> getreaders(){ 
    list<reader> readerlist = new arraylist<reader>(); 
    readerlist.add(new reader("shun1","123")); 
    readerlist.add(new reader("shun2","123")); 
    return readerlist; 
  } 
} 

 这两个类除了加入注解外,其他均和昨天讲的webservice的一样。这里就不多讲了,对注解的解释,大家可以看看javaee的文档。不过按意思应该很容易理解的。
接下来就是javabean,还是那个reader类:

public class reader{ 
  private static final long serialversionuid = 1l; 
  private string name; 
  private string password; 
   
  public reader(){} 
  public reader(string name,string password) { 
    this.name = name; 
    this.password = password; 
  } 
  //get/set方法省略 
  public string tostring(){ 
    return "name:"+name+",password:"+password; 
  } 
   
} 

 上面的已经写完了。
2)我们要用做web项目吗?不急,先不用,cxf自带了一个轻量的容器服务,相当于spring自己提供了ioc容器一样。我们可以先用它来测试一下我们部署成功没。
直接来一个测试类:

public static void main(string[] args) { 
    system.out.println("server is starting..."); 
    readerservice readerservice = new readerservice(); 
    endpoint.publish("http://localhost:8080/readerservice",readerservice); 
    system.out.println("server is started..."); 
  } 

 简单得不得了吧。直接publish地址,然后指定接口或类就ok了。我这里用的是类,但尽量用接口,毕竟面向接口编程才是真正的面对对象思想。
我们启动看看结果:

使用CXF和Jersey框架来进行Java的WebService编程

 我们看到启动已经完成,接着启动浏览器看看是否成功了。
直接在浏览器输入http://localhost:8080/readerservice?wsdl,我们可以看到:

使用CXF和Jersey框架来进行Java的WebService编程

它生成了我们所需要的wsdl文件,说明我们部署成功了。
3)部署成功后,我们就是要调用啦,它的调用也相当简单,跟xfire类似,取得接口,然后就可以跟本地类一样调用方法了。

public static void main(string[] args) { 
    jaxwsproxyfactorybean factorybean = new jaxwsproxyfactorybean(); 
    factorybean.setserviceclass(ireaderservice.class); 
    factorybean.setaddress("http://localhost:8080/readerservice"); 
     
    ireaderservice readerservice = (ireaderservice)factorybean.create(); 
    reader reader = readerservice.getreader("shun","123"); 
    system.out.println("reader:"+reader); 
  } 

 这里很简单,也是取得一个工厂类,然后直接设接口和地址再create就可以得取相应的接口了,这里跟xfire一样,也是需要调用端先定义好接口原型,否则这些调用将无从说起。
我们运行得到结果:

使用CXF和Jersey框架来进行Java的WebService编程

没问题,跟我们预想的结果一致。
 
4)但很多情况下,我们并不希望我们的webservice和我们的应用分开两个服务器,而希望他们在同一个容器,tomcat或jboss或其他的,这样我们就必须通过web来部署我们前面完成的webservice。
注意,我们这里需要用到spring定义文件。
首先看看web.xml:

<?xml version="1.0" encoding="utf-8"?> 
<web-app xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" 
  xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" 
  xsi:schemalocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" 
  id="webapp_id" version="3.0"> 
   
  <context-param> 
    <param-name>contextconfiglocation</param-name> 
    <param-value>web-inf/beans.xml</param-value> 
  </context-param> 
   
  <listener> 
    <listener-class>org.springframework.web.context.contextloaderlistener</listener-class> 
  </listener> 
   
  <servlet> 
    <servlet-name>cxfservlet</servlet-name> 
    <servlet-class>org.apache.cxf.transport.servlet.cxfservlet</servlet-class> 
  </servlet> 
  <servlet-mapping> 
    <servlet-name>cxfservlet</servlet-name> 
    <url-pattern>/webservice/*</url-pattern> 
  </servlet-mapping> 
</web-app> 

 这里很简单,只是指定了spring的监听器和相应的配置文件路径,并且指定了cxf的拦截方式。
接下来看看beans.xml:

<?xml version="1.0" encoding="utf-8"?> 
<beans xmlns="http://www.springframework.org/schema/beans" 
  xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" 
  xmlns:jaxws="http://cxf.apache.org/jaxws" 
  xsi:schemalocation=" 
    http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd 
    http://cxf.apache.org/jaxws 
    http://cxf.apache.org/schemas/jaxws.xsd"> 
  <import resource="classpath:meta-inf/cxf/cxf.xml" /> 
  <import resource="classpath:meta-inf/cxf/cxf-extension-soap.xml" /> 
  <import resource="classpath:meta-inf/cxf/cxf-servlet.xml" /> 
   
  <jaxws:endpoint id="readerservicce2" 
    implementor="com.cxf.servlet.readerservice" address="/readerservice2" /> 
</beans> 

 这里很简单,只是通过jaxws:endpoint定义了一个webservice,implementor是webservice的处理类,而address是它的访问路径,跟我们前面写的readerservice类似。
这时我们可以把它部署到tomcat中,通过http://localhost:8080/cxfwebservice/webservice/readerservice2?wsdl可以直接访问。
 
有些朋友会问,为什么这次访问的url跟前面的不一样呢。其实前面的访问地址是我们自己定义的,而这里的webservice地址是我们在配置文件中配置好的,并且是通过web项目来部署的,这里就需要用项目名称,而且我们在cxfservlet那里配置了url-pattern是webservice,所以最后的url就跟上面一致了。
我们可以看到效果:

使用CXF和Jersey框架来进行Java的WebService编程

 这证明我们部署成功了。
 
可以再次用前面的测试类测试一下,注意,需要把address修改成我们发布后的url。
cxf相比xfire又更简洁了一些,虽然它增加了一些注解,但这些无伤大雅,它只是把以前的services.xml中的信息集中到类中,反而更方便维护,但这还是见仁见智的,有些人就喜欢配置文件,而有些人就不喜欢。另外cxf的调用方式更加简洁,比起xfire它的代码量更小了,是一个较大的进步。
 
有些朋友在搭建的过程中出现了一些问题,免去一个个回复了,这里放出代码,有需要的朋友可以下载看看。
lib目录下的所有包均没有放入,把cxf的所有包放入即可。
注:所用ide为idea,文件结构跟eclipse不通用,如果需要在eclipse下使用的,可以直接复制代码和文件到eclipse新建的项目即可。

jersey
我们来看看它的基本用法。
 直接来个项目看看。开始项目之前大家还是先自己去下载包:https://maven.java.net/content/repositories/releases/com/sun/jersey/要运行例子需要同时下载server和client。当然不想找那么多,可以直接下这个zip包,https://maven.java.net/service/local/artifact/maven/redirect?r=releases&g=com.sun.jersey&a=jersey-archive&v=1.10&e=zip
 1)直接来个javabean

@xmlrootelement 
public class reader implements serializable{ 
  private static final long serialversionuid = 1l; 
  private string name; 
  private string password; 
   
  public reader(){} 
  public reader(string name,string password) { 
    this.name = name; 
    this.password = password; 
  } 
  //省略get/set方法 
  public string tostring(){ 
    return "name:"+name+",password:"+password; 
  } 
} 

  这里用到了一个标签,这里用到的表示它返回的时候的类型,即此reader类可以用于xml返回。
 2)来个service类,这时已经不用像以前的cxf和xfire一样要接口了,直接来个类就ok了。 

@path("/readerservice/{name}/{password}") 
public class readerservice { 
   
  @get 
  @produces(mediatype.application_json) 
  public reader getreader(@pathparam("name") string name,@pathparam("password") string password) { 
    return new reader(name,password); 
  } 
   
  public static void main(string[] args) throws illegalargumentexception, ioexception, urisyntaxexception { 
    httpserver server = httpserverfactory.create("http://localhost:8080/"); 
    server.start(); 
  } 
} 

  这时用到了几个标签,path相信用过springmvc的朋友应该知道这种写法,就是url匹配,如果不清楚的,可以先去看看。get标签表示这个方法只能通过get方法来进行访问,而produces表示生成的结果,它表示系统会把reader对象封闭成json结果进行返回。
 如果不理解不要紧,等一下看结果就可以理解了。
 而这时有一个main方法,相信有很大疑问吧。这是jersey内部提供的一个轻量级的内部容器,它可以暂时供我们调试用,但真正使用肯定不能用这个。
 3)我们写一个测试类

public class readerclient { 
 
  public static void main(string[] args) { 
    client client = client.create(); 
    webresource resource = client.resource("http://localhost:8080/readerservice/shun/123213"); 
    reader reader = resource.get(reader.class); 
    system.out.println(reader); 
  } 
 
} 

  很简单的代码,应该都看得懂的,一个client对象,请求webservice,返回一个resource,然后resource就直接调用相应的方法,当然这个方法是通过我们的url来进行匹配的。
 
 这里我们先用它自带的一个轻量级服务测试一下。直接运行readerservice,它里面有包含一个main方法,运行后,我们再运行readerclient,我们可以看到结果为:

使用CXF和Jersey框架来进行Java的WebService编程

  结果正确。
 
 我们当然不想就这样用自带的轻量级服务来作为我我们的服务器,我们需要放到和我们的项目在同一个服务器,比如tomcat,jboss等。
 4)web项目当然就少不了web.xml。

<?xml version="1.0" encoding="utf-8"?> 
<web-app xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" 
  xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" 
  xsi:schemalocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" 
  id="webapp_id" version="3.0"> 
  <servlet> 
    <servlet-name>jersey web application</servlet-name> 
    <servlet-class>com.sun.jersey.spi.container.servlet.servletcontainer</servlet-class> 
    <load-on-startup>1</load-on-startup> 
  </servlet> 
  <servlet-mapping> 
    <servlet-name>jersey web application</servlet-name> 
    <url-pattern>/rest/*</url-pattern> 
  </servlet-mapping> 
  <welcome-file-list> 
    <welcome-file>index.jsp</welcome-file> 
  </welcome-file-list> 
</web-app> 

  这时指定了rest路径下的所有都将被jersey进行拦截。
 我们部署到tomcat下启动后再重新运行readerclient,注意先要修改resource的路径:

webresource resource = client.resource("http://localhost:8080/jerseywebservice/rest/readerservice/shun/123213"); 

  我的项目名为jerseywebservice,请根据你的项目名进行修改。
 修改后,我们重新运行,结果如下:

使用CXF和Jersey框架来进行Java的WebService编程

和上面的结果一致,说明部署的效果是一样的,也是正确的。