Spring HTTP invoker是spring框架中的一个远程调用模型,执行基于HTTP的远程调用(意味着可以通过防火墙),并使用java的序列化机制在网络间传递对象。客户端可以很轻松的像调用本地对象一样调用远程服务器上的对象,这有点类似于webservice,但又不同于webservice,区别如下:
webservice |
HTTP invoker |
跨平台,跨语言 |
只支持java语言 |
支持SOAP,提供wsdl |
不支持 |
结构庞大,依赖特定的webservice实现,如xfire等 |
结构简单,只依赖于spring框架本身 |
项目中使用哪种远程调用机制取决于项目本身的要求。
²HTTP invoker服务模式
说明:
1.服务器端:通过HTTP invoker服务将服务接口的某个实现类提供为远程服务
2.客户端:通过HTTP invoker代理向服务器端发送请求,远程调用服务接口的方法
3.服务器端与客户端通信的数据需要序列化
1.添加springJAR文件
建议使用spring2+.jar版本
2.创建服务接口
3.创建服务接口的具体实现类
4.公开服务
1.添加springJAR文件
建议使用spring2+.jar版本
2.创建服务接口
3.访问服务
1.服务接口:UcService.java
它提供两项服务,查询用户信息和记录日志,如下:
publicinterfaceUcService {
publicUserInfo getUserInfobyName(String userName);
publicintrecordLog(String username, String point, String operate, String desc);
}
说明:举这个列子是因为其比较有代表性,它将展示普通数据类型(int,long等)和复杂数据类型(DTO等)的远程调用方式。UserInfo是一个普通的DTO,代码如下:
publicclassUserInfoimplementsSerializable {
privatestaticfinallongserialVersionUID= -6970967506712260305L;
/**用户名*/
privateStringuserName;
/**电子邮箱*/
privateStringemail;
/**注册日期*/
privateDateregistDate;
publicString getUserName() {
returnuserName;
}
publicvoidsetUserName(String userName) {
this.userName= userName;
}
publicString getEmail() {
returnemail;
}
publicvoidsetEmail(String email) {
this.email= email;
}
publicDate getRegistDate() {
returnregistDate;
}
publicvoidsetRegistDate(Date registDate) {
this.registDate= registDate;
}
}
注意:因为是在网络间传输对象,所以需要将UserInfo实现Serializable接口,并指定一个serialVersionUID(任意值即可,同时客户端也要有这个类,否则在客户端接收对象时会因为serialVersionUID不匹配而出现异常)
回到UcService.java,它提供了两个服务(在这里一个方法代表一个服务功能),我们需要具体的实现类来实现真正的服务
2.实现类是UCServiceImpl.java
publicclassUCServiceImplimplementsUcService {
privatestaticLoggerpointrecordlog= Logger.getLogger("pointrecordlog");
privatestaticLoggerlogger= Logger.getLogger(UCServiceImpl.class);
privateUcFacadeucFacade;
publicvoidsetUcFacade(UcFacade ucFacade) {
this.ucFacade= ucFacade;
}
publicUserInfo getUserInfobyName(String userName) {
UserInfo user =null;
try{
user =ucFacade.getUserInfoDetail(userName);
logger.debug("get userinfo success byusername:"+ userName);
}catch(Throwable t) {
logger.error("get userinfo fail byusername:"+ userName, t);
}
returnuser;
}
publicintrecordLog(String username, String point, String operate, String desc) {
intresult = 0;
try{
pointrecordlog.info(username +" - "+ point +" - "+ operate +" - "+ desc);
}catch(Throwable t) {
result = -1;
logger.error(t);
}
returnresult;
}
}
说明:ucFacade是通过spring注入的一个数据查询类,因为它与http invoker没有直接关系,所以不进行介绍。
3.公开服务UcService.java
²WEB-INF/application-context.xml:将接口声明为HTTP invoker服务
<beanid="httpService"
class="org.springframework.remoting.httpinvoker.HttpInvokerServiceExporter">
<propertyname="service">
<refbean="ucService"/>
</property>
<propertyname="serviceInterface"
value="com.netqin.baike.service.UcService">
</property>
</bean>
<beanid="ucService"class="com.netqin.baike.service.impl.UCServiceImpl"/>
说明:HttpInvokerServiceExporter实际上是一个spring mvc控制器,它处理客户端的请求并调用服务实现。
²WEB-INF/service-servlet.xml:HttpInvokerServiceExporter实际上是一个spring mvc控制器,所以需要为其提供spring URL处理器,这里我们使用SimpleUrlHandlerMapping
<bean
class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
<propertyname="mappings">
<props>
<propkey="/httpService">httpService</prop>
</props>
</property>
</bean>
²WEB-INF/web.xml:配置spring监听及DispatcherServlet
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
/WEB-INF/application-context.xml
</param-value>
</context-param>
<listener>
<listener-class>
org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
<servlet>
<servlet-name>service