WebService
一.WebService概述
1.1WebService简介
Web Service(WEB服务)能够快捷和方便地综合结合各种系统、商务和任何应用平台。利用最新的Web Service 标准能够使任何软件系统和系统之间的应用互通互联,方便,而且更加廉价。
1.2WebService的应用场景
跨越防火墙通信
集成应用程序
复用软件
1.3其他远程应用调用解决方案
使用Socket远程通信
使用Apache的HttpClient
RMI(Remote Method Invoke)
1.4WebService的三个规范
JAXM&SAAJ(废弃)
JAX-WS :采用标准SOAP(Simple Object Access Protocol) 协议传输,soap协议传输是xml数据。
JAX-RS:Java针对REST(Representation State Transfer)风格制定的一套Web服务规范。
二.Apache的CXF
Apache CXF是一个开源的Web Service 框架,CXF 帮助我们构建和开发 Web Service,它支持多种协议,如:SOAP, XML/HTTP、RESTful或者CORBA。
官网:http://cxf.apache.org/
下载地址:http://cxf.apache.org/download.html
三.基于JAX-WS规范的入门
3.1JAX-WS的三要素
SOAP:基于HTTP协议,采用XML格式,用来传递信息的格式。
WSDL:用来描述如何访问具体的服务
UDDI:用户可按UDDI标准搭建UDDI服务器,用来管理分发,查询WebService。其他用户可以自己注册发布WebService调用。
3.2入门案例
我们在使用WebService时,更多的时候充当的是调用者。因为服务端别人已经写好了。
3.2.1编写服务端
第一步:创建Maven工程导入CXF坐标
<!-- 导入cxf相关坐标 -->
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-frontend-jaxws</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.25</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-features-logging</artifactId>
<version>3.3.0</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-http-jetty</artifactId>
<version>3.3.0</version>
</dependency>
第二步:编写服务接口和实现类
/**
* 提供实时公交的服务接口
* @author Mr.song
* @date 2019/05/20 9:55
*/
@WebService //表明当前接口是一个WebService服务
public interface BusService {
/**
* 根据公交线路查询公交实时站点
* @param line 公交线路
* @return
*/
String getBusByLine(String line);
}
//==========================================
/**提供实时公交的服务实现类
* @author Mr.song
* @date 2019/05/20 9:57
*/
public class BusServiceImpl implements BusService {
public String getBusByLine(String line) {
if ("916".equals(line)){
return "距您还有3站";
}
return "未知";
}
}
Tips:注意加入@WebService注解
第三步:编写测试类(main方法启动)
/**
* @author Mr.song
* @date 2019/05/20 10:17
*/
public class WebBusServiceTest {
public static void main(String[] args) {
//1.创建发布服务的对象
JaxWsServerFactoryBean factoryBean = new JaxWsServerFactoryBean();
//2.设置服务地址
factoryBean.setAddress("http://localhost:8847/bus");
//3.设置服务的对象
factoryBean.setServiceBean(new BusServiceImpl());
//4.发布服务
factoryBean.create();
}
}
第四步:浏览器访问测试
地址为:http://localhost:8847/bus/?wsdl
若控制台无错误,则表明服务端编写正确。
3.2.2编写客户端
第一步:创建Maven工程导入CXF坐标(同上)
第二步:使用jdk提供的命令生成本地代码
使用的是jdk bin目录下的wsimport.exe工具
进入到客户端工程java目录下,按住shift右击,在此处打开shell窗口输入命令:wsimport -s . http://localhost:8847/bus/?wsdl 即可完成解析,并生成代码。(此处的链接是服务端浏览器测试时访问的)| -s 指定生成文件目录 . 指当前文件夹下。
第三步:编写测试类
/**
* @author Mr.song
* @date 2019/05/20 16:32
*/
public class BusTest {
@Test
public void testBus(){
//1.创建客户端的代理工厂bean
JaxWsProxyFactoryBean proxy = new JaxWsProxyFactoryBean();
//2.设置说明书的地址
proxy.setAddress("http://localhost:8847/bus?wsdl");
//3.设置客户端需要使用的服务接口类字节码
proxy.setServiceClass(BusService.class);
//4.使用proxy创建服务接口的代理对象
Object obj = proxy.create();
//5.如果obj是service接口的代理实现类,强转成service接口
BusService busService = (BusService) obj;
//6.执行service中的方法
String rtValue = busService.getBusByLine("916");
System.out.println(rtValue);
}
}
Tips: 运行测试端代码时,保证服务端不要关闭
3.3Spring整合CXF
3.3.1编写服务端
第一步:创建Maven的Web工程并导入CXF坐标(同上)
第二步:编写服务接口和实现类(同上)
第三步:配置 Spring 的配置文件
<?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">
<!-- 配置服务端的服务 -->
<!-- 主机和端口用tomcat的,http协议 -->
<jaxws:server address="/bus">
<!-- 配置具体服务实现类 -->
<jaxws:serviceBean>
<bean class="cn.dintalk.service.BusServiceImpl"/>
</jaxws:serviceBean>
</jaxws:server>
</beans>
第四步:在 web.xml 中配置Spring容器在应用加载时创建
<?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"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.5">
<!-- 配置应用加载监听器 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext-cxf-server.xml</param-value>
</context-param>
<!-- CXF框架的核心控制器:CXFServlet -->
<servlet>
<servlet-name>cxf</servlet-name>
<servlet-class>org.apache.cxf.transport.servlet.CXFServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>cxf</servlet-name>
<!-- 配置处理请求url中含有/ws/*的请求 -->
<url-pattern>/ws/*</url-pattern>
</servlet-mapping>
</web-app>
第五步:浏览器访问测试
地址:http://localhost:8080/ws/bus?wsdl
若控制台无错误,则表明服务端编写正确。
3.3.2编写客户端
第一步:创建Maven的Web工程并导入CXF坐标(同上)
第二步:使用 jdk 提供的命令生成客户端代码(同上)
第三步:编写 Spring 配置文件
<?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">
<!-- 配置服务端访问地址和应用接口 -->
<jaxws:client id="client"
address="http://localhost:8080/ws/bus?wsdl"
serviceClass="cn.dintalk.service.BusService"></jaxws:client>
</beans>
第四步:编写测试类
/**
* @author Mr.song
* @date 2019/05/20 17:29
*/
public class BusTest {
@Test
public void testBus(){
//1.读取配置文件,创建容器
ApplicationContext ac = new
ClassPathXmlApplicationContext("classpath:applicationContext-cxf-client.xml");
//2.根据bean的id获取对象
BusService busService = (BusService) ac.getBean("client");
//3.执行方法
String rtValue = busService.getBusByLine("916");
System.out.println(rtValue);
}
}
Tips: 运行测试类测试时,主要不要关闭服务端。
四.基于JAX-RS规范的入门
4.1JAX-RS和Restful
JAX-RS 是一个 Java 编程语言接口,被设计用来简化使用 REST 架构的应用程序的开发。JAX-RS API 使用 Java 编程语言的注解来简化 RESTful web service 的开发。开发人员使用 JAX-RS 的注解修饰Java编程语言的类文件来定义资源和能够应用在资源上的行为。JAX-RS的注解是运行时的注解,因此运行时的映射会为资源生成辅助类和其他的辅助文件。包含 JAX-RS 资源类的 Java EE 应用程序中资源是被配置好的,辅助类和辅助文件是生成的,资源通过被发布到 Java EE 服务器上来公开给客户端。
RESTful web service 是创建来能在 web 更好的运行的 web service。REST 是一种架构类型,指定了如统一的接口等应用于 web service 的约束。REST 提供了如性能、可扩展性和可变性等特性,使得 service 能够更好的在web上工作。在 REST 框架中,数据和功能被认为是资源,是通过 URI来访问的,通常是 web链接。资源是通过使用一组简单的、定义良好的操作来生效。REST 的架构方式限定了客户/服务器架构,是设计来使用无状态的通信协议的,通常是 HTTP。在 REST 框架类型中,客户端和服务器使用标准的接口和协议交换资源的representation。
4.2入门案例
4.2.1编写服务端
第一步:创建Maven工程并导入坐标
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-frontend-jaxrs</artifactId>
<version>3.0.1</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-transports-http-jetty</artifactId>
<version>3.0.1</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.7.12</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-rs-client</artifactId>
<version>3.0.1</version>
</dependency>
<dependency>
<groupId>org.apache.cxf</groupId>
<artifactId>cxf-rt-rs-extension-providers</artifactId>
<version>3.0.1</version>
</dependency>
<dependency>
<groupId>org.codehaus.jettison</groupId>
<artifactId>jettison</artifactId>
<version>1.3.7</version>
</dependency>
第二步:编写实体类(业务实体)
@XmlRootElement(name = "User")
public class User {
private Integer id;
private String username;
private String city;
private List<Car> cars = new ArrayList<Car>();
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
public List<Car> getCars() {
return cars;
}
public void setCars(List<Car> cars) {
this.cars = cars;
}
@Override
public String toString() {
return "User [id=" + id + ", username=" + username + ", city=" + city + ", cars=" + cars + "]";
}
@XmlRootElement(name = "Car")
public class Car {
private Integer id;
private String carName;
private Double price;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getCarName() {
return carName;
}
public void setCarName(String carName) {
this.carName = carName;
}
public Double getPrice() {
return price;
}
public void setPrice(Double price) {
this.price = price;
}
@Override
public String toString() {
return "Car [id=" + id + ", carName=" + carName + ", price=" + price + "]";
}
}
第三步:编写业务接口和实现类
@Path("/userService")
@Produces("*/*")
public interface UserService {
@POST
@Path("/user")
@Consumes({ "application/xml", "application/json" })
public void saveUser(User user);
@PUT
@Path("/user")
@Consumes({ "application/xml", "application/json" })
public void updateUser(User user);
@GET
@Path("/user")
@Produces({ "application/xml", "application/json" })
public List<User> findAllUsers();
@GET
@Path("/user/{id}")
@Consumes("application/xml")
@Produces({ "application/xml", "application/json" })
public User finUserById(@PathParam("id") Integer id);
@DELETE
@Path("/user/{id}")
@Consumes({"application/xml", "application/json"})
public void deleteUser(@PathParam("id") Integer id);
}
/**
* @author Mr.song
* @date 2019/05/20 19:39
*/
public class UserServiceImpl implements UserService {
public void saveUser(User user) {
System.out.println("save user:" + user);
}
public void updateUser(User user) {
System.out.println("update user:" + user);
}
public List<User> findAllUsers() {
List<User> users = new ArrayList<User>();
User user1 = new User();
user1.setId(1);
user1.setUsername("小明");
user1.setCity("北京");
List<Car> carList1 = new ArrayList<Car>();
Car car1 = new Car();
car1.setId(101);
car1.setCarName("保时捷");
car1.setPrice(1000000d);
carList1.add(car1);
Car car2 = new Car();
car2.setId(102);
car2.setCarName("宝马");
car2.setPrice(400000d);
carList1.add(car2);
user1.setCars(carList1);
users.add(user1);
User user2 = new User();
user2.setId(2);
user2.setUsername("小丽");
user2.setCity("上海");
users.add(user2);
return users;
}
public User finUserById(Integer id) {
if (id == 1) {
User user1 = new User();
user1.setId(1);
user1.setUsername("小明");
user1.setCity("北京");
return user1;
}
return null;
}
public void deleteUser(Integer id) {
System.out.println("delete user id :" + id);
}
}
第四步:编写服务发布类
/**
* jaxrs服务的发布
* @author Mr.song
* @date 2019/05/20 19:45
*/
public class UserTest {
public static void main(String[] args) {
//1.创建jaxrs服务工厂bean对象
JAXRSServerFactoryBean factoryBean = new JAXRSServerFactoryBean();
//2.设置服务地址
factoryBean.setAddress("http://localhost:8848");
//3.设置提供服务的类
factoryBean.setServiceBean(new UserServiceImpl());
//4.设置日志输入和输出的两个拦截器
factoryBean.getInInterceptors().add(new LoggingInInterceptor());
factoryBean.getOutInterceptors().add(new LoggingOutInterceptor());
//5.发布服务
factoryBean.create();
}
}
4.2.2编写客户端
第一步:创建Maven工程并导入坐标(同时)
第二步:编写业务实体类(同上)
第三步:编写测试类
/**
* @author Mr.song
* @date 2019/05/20 19:55
*/
public class UserClientTest {
/**
* 注意:它和jaxws规范的区别是,它使用的是restful的url
*/
@Test
public void test(){
User user = new User();
user.setId(1);
user.setUsername("test");
//1.使用WebClient创建访问地址 : 在服务端接口中有/userService /user的路径规则
Object rtValue = WebClient.create("http://localhost:8848/userService/user")
.type(MediaType.APPLICATION_XML).post(user);
System.out.println(rtValue);
}
@Test
public void test2(){
//1.使用WebClient创建访问地址
Object rtValue = WebClient.create("http://localhost:8848/userService/user/123")
.type(MediaType.APPLICATION_JSON).delete();
System.out.println(rtValue);
}
}
Tips: 测试时保证服务端开启。
4.3Spring整合CXF
4.3.1编写服务端
第一步:创建Maven的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"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.5">
<!-- 配置应用加载监听器 -->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext-cxf-rs.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!-- CXF的核心控制器 -->
<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>/rs/*</url-pattern>
</servlet-mapping>
</web-app>
第四步:编写Spring配置文件
<?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:jaxrs="http://cxf.apache.org/jaxrs"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://cxf.apache.org/jaxrs
http://cxf.apache.org/schemas/jaxrs.xsd">
<!-- 配置服务:ip、端口用tomcat的 -->
<!-- 这里配置了address,接口类就不要配置注解了(@Path("/userService")) -->
<jaxrs:server id="userService" address="/userService">
<jaxrs:serviceBeans>
<bean class="cn.dintalk.service.UserServiceImpl"/>
</jaxrs:serviceBeans>
<!-- 配置拦截器 -->
<jaxrs:inInterceptors>
<bean class="org.apache.cxf.interceptor.LoggingInInterceptor"/>
</jaxrs:inInterceptors>
<jaxrs:outInterceptors>
<bean class="org.apache.cxf.interceptor.LoggingOutInterceptor"/>
</jaxrs:outInterceptors>
</jaxrs:server>
</beans>
4.3.2编写客户端
第一步:创建Maven工程并导入坐标(同上)
第二步:创建业务实体类(同上)
第三步:编写测试类
@Test
public void test04(){
User user = WebClient.create("http://localhost:8080/rs/userService/user/1")
.type(MediaType.APPLICATION_XML).get(User.class);
System.out.println(user);
}
//输出如下:
//User [id=1, username=小明, city=北京, cars=[]]
Tips: 测试时要保证服务端开启。