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文件的内容,为元素加一个scope属性,该属性 有四个可取的值:Application, SOAPSession, TransportSession, Request,不过要注意一下,虽然 Axis2的官方文档将这四个值的单词首字母和缩写字母都写成了大写,但经笔者测试,必须全部小写才有 效,也就是这四个值应为:application、soapsession、transportsession、request,其中request为 scope属性的默认值。读者可以选择使用transportsession和application分别实现同一个WebService类和 跨WebService类的会话管理。

在客户端需要使用setManageSession(true)打开Session管理功能。

综上所述,实现同一个WebService的Session管理需要如下三步:

1.使用MessageContext和ServiceContext获得与设置key-value对。

2.为要进行Session管理的WebService类所对应的元素添加一个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):



登录服务 service.LoginService



%AXIS2_HOME%\bin\wsdl2java -uri 
http://localhost:8080/axis2/services/loginService? wsdl -p client -s -o 
stub


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);
}



// 第3步:打开客户端的Session管理功能

_serviceClient.getOptions().setManageSession(true);
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);

}