servletAPI在Struts2中的角色
Xwork与Web是无关的,Action不用去依赖于任何Web容器,不用和那些 JavaServlet复杂的请求(Request)、响应(Response)关联在一起。对请求(Request)的参数(Param),可以使用拦截器框架自动调用一些get()和set()方法设置(注入)到对应Action的字段中。但是,仅仅取得请求参数的值就能完全满足的功能要求吗?不,在 Web应用程序开发中,除了将请求参数自动设置到Action的字段中,往往也需要在Action里直接获取请求(Request)或会话 (Session)的一些信息,甚至需要直接对JavaServlet Http的请求(HttpServletRequest)、响应(HttpServletResponse)操作。
ActionContext(com.opensymphony.xwork.ActionContext)
可以通过ActionContext的静态方法:ActionContext.getContext()来取得当前的ActionContext对象
ActionContext是Action执行时的上下文,上下文 可以看作是一个容器(这里的容器是一个Map),它存放放的是Action在执行时需要用到的对象,比如:在使用WebWork时,上下文放有请求的参数(Parameter)、会话(Session)、Servlet上下文(ServletContext)、本地化(Locale) 信息等。
WebWork框架将与Web相关的很多对象重新进行了包装,将HttpSession对象重新包装成了一个Map对象,供Action使用,而不用直接和底层的HttpSession打交道。也正是框架 的包装,让的Actoion可以完全的和Web层解藕。
ServletActionContext(com.opensymphony.webwork. ServletActionContext)
这个类直接继承了上面介绍的ActionContext,它提供了直接与JavaServlet相关对象访问的功能,它可以取得的对象有:
1、javax.servlet.http.HttpServletRequest:HTTPservlet请求对象
2、javax.servlet.http.HttpServletResponse;:HTTPservlet相应对象
3、javax.servlet.ServletContext:Servlet 上下文信息
4、javax.servlet.ServletConfig:Servlet配置对象
5、javax.servlet.jsp.PageContext:Http页面上下文
ServletActionContext除了提供了上面这些对象访问,它当然也继承了它父类ActionContex的很多功能,比如:对OgnlValueStack、Action名字等的访问。
二者的选择
ServletActionContext和ActionContext有着一些重复的功能,在Action中,该如何去抉择呢?遵循的 原则是:如果ActionContext能够实现的功能,那最好就不要使用ServletActionContext,让的Action尽量不要 直接去访问JavaServlet的相关对象。
在使用ActionContext时有一点要注意:不要在Action的构造函数里使用 ActionContext.getContext(),因为这个时候ActionContext里的一些值也许没有设置,这时通过 ActionContext取得的值也许是null。
他们都是线程安全的
在每次执行Action之前都会创建新的ActionContext,ActionContext是线程安全的,也就是说在同一个线程里ActionContext里的属性是唯一的,这样Action就可以在多线程中使用。
看看这段代码:
public static ActionContext getContext() {
ActionContext context = (ActionContext) actionContext.get();
if (context == null) {
OgnlValueStack vs = new OgnlValueStack();
context = new ActionContext(vs.getContext());
setContext(context);
}
return context;
}
一般情况,ActionContext都是通过:ActionContext context = (ActionContext) actionContext.get();来获取的。再来看看这里的actionContext对象的创建:static ThreadLocal actionContext = new ActionContextThreadLocal();,ActionContextThreadLocal是实现ThreadLocal的一个内部 类。ThreadLocal可以命名为“线程局部变量”,它为每一个使用该变量的线程都提供一个变量值的副本,使每一个线程都可以独立地改变自己的副本, 而不会和其它线程的副本冲突。这样,ActionContext里的属性只会在对应的当前请求线程中可见,从而保证它是线程安全的。
关于线程局部变量请参看另一篇文章《线程局部变量》