WebService之Axis2系列教程(五)会话(Session)管理
WebService给人最直观的感觉就是由一个个方法组成,并在客户端通过SOAP协议调用这些方法。这些方法可能有返回值,也可能没有返回值。虽然这样可以完成一些工具,但这些被调用的方法是孤立的,当一个方法被调用后,在其他的方法中无法获得这个方法调用后的状态,也就是说无法保留状态。 ? 读者可以想象,这对于一个完整的应用程序,无法保留状态,就意味着只依靠WebService很难完成全部的工作。例如,一个完整的应用系统都需要进行登录,这在Web应用中使用Session来保存用户登录状态,而如果用WebService的方法来进行登录处理,无法保存登录状态是非常令人尴尬的。当然,这也可以通过其他的方法来解决,如在服务端使用static变量来保存用户状态,并发送一个id到客户端,通过在服务端和客户端传递这个id来取得相应的用户状态。这非常类似于Web应用中通过Session和Cookie来管理用户状态。但这就需要由开发人员做很多工作,不过幸好Axis2为我们提供了WebService状态管理的功能。 ? 使用Axis2来管理WebService的状态基本上对于开发人员是透明的。在WebService类需要使用org.apache.axis2.context.MessageContext和org.apache.axis2.context.ServiceContext类来保存与获得保存在服务端的状态信息,这些对象使用HttpSession接口的getAttribute和setAttribute方法获得与设置Session域属性。 ? 除此之外,还需要修改services.xml文件的内容,为<service>元素加一个scope属性,该属性有四个可取的值:Application,SOAPSession,TransportSession,Request,不过要注意一下,虽然Axis2的官方文档将这四个值的单词首字母和缩写字母都写成了大写,但经笔者测试,必须全部小写才有效,也就是这四个值应为:application、soapsession、transportsession、request,其中request为scope属性的默认值。读者可以选择使用transportsession和application分别实现同一个WebService类和跨WebService类的会话管理。 ? Web服务有着很大的需求,很多人进入了Web服务这一领域,其结果是人们需要Web服务具有更多的特性,以便使用Web服务可以完成任何事情。但Web服务从设计上来说是没有状态的。在Web服务世界起初的时候是没有管理会话的概念的,但是现在形势改变了:没有会话管理,开发者是无法开发出高级的应用程序的。一个很好的例子是银行系统,您登录进入系统,然后取钱,最后退出登录。
?
public void init(ServiceContext serviceContext) { } public void destroy(ServiceContext serviceContext) { } ??? 除此之外,当服务接收到一个请求时,它会传入相应的操作上下文作为参数进行通知。如果服务想访问输入消息的上下文或输入的SOAP消息,可以在服务的实现类中加入以下方法来做到这一点,在真正的方法调用之前调用这一方法。? public void setOperationContext(OperationContext operationContext) { } ??? ??? 请求会话范围(Request Session Scope)
<wsa:ReplyTo> <wsa:Address> http://www.w3.org/2005/08/addressing/anonymous </wsa:Address> <wsa:ReferenceParameters> <axis2:ServiceGroupId xmlns:axis2= "http://ws.apache.org/namespaces/axis2"> urn:uuid:65E9C56F702A398A8B11513011677354 </axis2:ServiceGroupId> </wsa:ReferenceParameters> </wsa:ReplyTo> ?? 如果客户端想保持在同一个会话内,它要复制那个引用参数,并在第二次调用服务的时候传回服务器。只要客户端传输了正确的“serviceGroupId”,它就会在同一个会话内,服务也可以维护会话相关的数据。与请求会话不同,SOAP有默认的失效期,如果客户端超过30秒没有连接服务,会话就会过期。如果客户端发送了过期的“serviceGroupId”,会得到Axis错误。 <service name="foo" scope="soapsession"> </service>
<service name="foo" scope="transportsession"> </service> ????如果打算使用Axis2,以传输会话方式部署服务需要对axis2.xml做些修改。这主要是为了提高内存的使用率;否则,无论是否以传输会话方式部署服务,Axis2都会在传输的级别建立会话对象;有了这些变化,Axis就不会创建无用的对象了。为了管理传输级别的会话,需要在axis2.xml中设置“manageTransportSession”参数为“true”:? <parameter name="manageTransportSession" locked="false">true</parameter>
<service name="foo" scope="application"> </service>
Options options = new Options(); options.setManageSession(true); ServiceClient sender = new ServiceClient(); sender.setOptions(options);
? 在客户端需要使用setManageSession(true)打开Session管理功能。 综上所述,实现同一个WebService的Session管理需要如下三步: 1.?使用MessageContext和ServiceContext获得与设置key-value对。 2.?为要进行Session管理的WebService类所对应的<service>元素添加一个scope属性,并将该属性值设为transportsession。 3.?在客户端使用setManageSession(true)打开Session管理功能。 下面是一个在同一个WebService类中管理Session的例子。 先建立一个WebService类,代码如下: package service; import org.apache.axis2.context.ServiceContext; import org.apache.axis2.context.MessageContext; public class LoginService { public boolean login(String username,String password) { if("bill".equals(username) && "1234".equals(password)) { // 第1步:设置key-value对 MessageContext mc = MessageContext.getCurrentMessageContext(); ServiceContext sc = mc.getServiceContext(); sc.setProperty("login","成功登录"); return true; } else { return false; } } public String getLoginMsg() { // 第1步:获得key-value对中的value MessageContext mc = MessageContext.getCurrentMessageContext(); ServiceContext sc = mc.getServiceContext(); return (String)sc.getProperty("login"); } } 在LoginService类中有两个方法:login和getLoginMsg,如果login方法登录成功,会将“成功登录”字符串保存在ServiceContext对象中。如果在login方法返回true后调用getLoginMsg方法,就会返回“成功登录”。 ????下面是LoginService类的配置代码(services.xml): <!-- 第2步:添加scope属性 --> <service name="loginService" scope="transportsession"> <description> 登录服务 </description> <parameter name="ServiceClass"> service.LoginService </parameter> <messageReceivers> <messageReceiver mep="http://www.w3.org/2004/08/wsdl/in-out" class="org.apache.axis2.rpc.receivers.RPCMessageReceiver" /> </messageReceivers> </service> 使用如下的命令生成客户端使用的stub类: %AXIS2_HOME%binwsdl2java -uri http://localhost:8080/axis2/services/loginService?wsdl -p client -s -o stub %AXIS2_HOME%binwsdl2java -uri http://localhost:8080/axis2/services/loginService?wsdl -p client -s -o D:/Work/NPU/Develop/eclipse-workspace/SayHello/src ????在stubsrcclient目录中生成了一个LoginServiceStub.java类,在该类中找到如下的构造句方法: public LoginServiceStub( org.apache.axis2.context.ConfigurationContext configurationContext,java.lang.String targetEndpoint,boolean useSeparateListener) throws org.apache.axis2.AxisFault { _serviceClient.getOptions().setSoapVersionURI( org.apache.axiom.soap.SOAP12Constants.SOAP_ENVELOPE_NAMESPACE_URI); } 在该方法中最后添加如下的代码(在Axis1.5以上版本中不需要这么做): // 第3步:打开客户端的Session管理功能 _serviceClient.getOptions().setManageSession(true); 下面的客户端代码使用LoginServiceStub对象访问了刚才建立的WebService: LoginServiceStub stub = new LoginServiceStub(); LoginServiceStub.Login login = new LoginServiceStub.Login(); login.setUsername("bill"); login.setPassword("1234"); if(stub.login(login).local_return) { System.out.println(stub.getLoginMsg().local_return); } 运行上面的代码后,会输出“成功登录”信息。 ? ? ? 教程源码下载: AxisTest AxisSpring AxisProject 转载请标明出处http://blog.csdn.net/shimiso 欢迎有识之士加入我们的技术交流群:173711587 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |