首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 软件管理 > 软件架构设计 >

Strust组件—Action种详解

2012-11-05 
Strust组件—Action类详解Action类是用户请求和业务逻辑之间的桥梁,每个Action充当客户的一项业务代理。在Re

Strust组件—Action类详解
Action类是用户请求和业务逻辑之间的桥梁,每个Action充当客户的一项业务代理。在RequestProcessor类预处理请求时,在创建了Action的实例后,就调用自身的processActionPerform()方法,该方法在调用Action类的execute()。
Action的excute()方法调用模型的业务方法,完成用户请求,然后根据执行结果把请求转发给其他合适的WEB组件。

一、Action类缓存

struts应用的生命周期中RequestProcessor只保证一个Action实例,所有的客户请求都共享这个实例.所有请求可以同时执行它的excute()方法。RequestProcessor类包含一个HashMap,作为存放所有Action实例的缓存。每个Action实例在缓存中存放的key为Action类名。在RequestProcessor类的processActionCreate()方法中,首先检查在HashMap中是否存在Action实例,如果有直接使用,否则创建一个新的。创建Action实力的代码位于同步代码块中,以保证只有一个线程创建Action实例,然后放在HashMap中。供其他线程使用。
如下代码

代码
protected Action processActionCreate(HttpServletRequest request,   
                                       HttpServletResponse response,   
                                       ActionMapping mapping)   
      throws IOException {   
  
      // Acquire the Action instance we will be using (if there is one)   
      String className = mapping.getType();   
      if (log.isDebugEnabled()) {   
          log.debug(" Looking for Action instance for class " + className);   
      }   
  
      // :TODO: If there were a mapping property indicating whether   
      // an Action were a singleton or not ([true]),   
      // could we just instantiate and return a new instance here?   
  
      Action instance = null;   
      synchronized (actions) {   
  
          // Return any existing Action instance of this class   
          instance = (Action) actions.get(className);   
          if (instance != null) {   
              if (log.isTraceEnabled()) {   
                  log.trace("  Returning existing Action instance");   
              }   
              return (instance);   
          }   
  
          // Create and return a new Action instance   
          if (log.isTraceEnabled()) {   
              log.trace("  Creating new Action instance");   
          }   
             
          try {   
              instance = (Action) RequestUtils.applicationInstance(className);   
              // :TODO: Maybe we should propagate this exception   
              // instead of returning null.   
          } catch (Exception e) {   
              log.error(   
                  getInternal().getMessage("actionCreate", mapping.getPath()),   
                  e);   
                     
              response.sendError(   
                  HttpServletResponse.SC_INTERNAL_SERVER_ERROR,   
                  getInternal().getMessage("actionCreate", mapping.getPath()));   
                     
              return (null);   
          }   
             
          instance.setServlet(this.servlet);   
          actions.put(className, instance);   
      }   
  
      return (instance);   
  
  }  


二.创建支持多线程的Action
1.什么是线程安全的代码
在多线程环境下能正确执行的代码就是线程安全的。
安全的意思是能正确执行,否则后果是程序执行错误,可能出现各种异常情况。

2.如何编写线程安全的代码
很多书籍里都详细讲解了如何这方面的问题,他们主要讲解的是如何同步线程对共享资源的使用的问题。主要是对synchronized关键字的各种用法,以及锁的概念。
Java1.5中也提供了如读写锁这类的工具类。这些都需要较高的技巧,而且相对难于调试。

但是,线程同步是不得以的方法,是比较复杂的,而且会带来性能的损失。等效的代码中,不需要同步在编写容易度和性能上会更好些。
我这里强调的是什么代码是始终为线程安全的、是不需要同步的。如下:
1)常量始终是线程安全的,因为只存在读操作。
2)对构造器的访问(new 操作)是线程安全的,因为每次都新建一个实例,不会访问共享的资源。
3)最重要的是:局部变量是线程安全的。因为每执行一个方法,都会在独立的空间创建局部变量,它不是共享的资源。局部变量包括方法的参数变量。

Servlet是在多线程环境下的。即可能有多个请求发给一个servelt实例,每个请求是一个线程。 struts下的action也类似,同样在多线程环境下,你也必须编写线程安全的Action类。
保证线程安全的原则就是仅仅使用局部变量,谨慎使用实例变量(拥有状态的实例,尤其是拥有业务对象状态的实例). 如果要用到那些有状态的实例,唯一和最好的办法是在Action类中,仅仅在Action类的execute()方法中使用局部变量,对于每个调用execute()方法的线程,JVM会在每个线程的堆栈中创建局部变量,因此每个线程拥有独立的局部变量,不会被其他线程共享.当线程执行完execute()方法后,它的局部变量就会被销毁.
如果Action类的实例变量是必须的话,需要采用JAVA同步机制(synchronized)对访问共享资源的代码块进行同步

三、Struts的几种Action
Struts提供了一些现成的Action类,直接使用可以大大节省时间,如下
ForwardAction
可以转发到其他web组件,仅仅提供一个转发功能,不作处理。
IncludeAction
包含其他web组件。
DiapatchAction
通常一个Action只完成一个操作,用这个Action可以完成一组相关的操作。
LookupDispatchAction
他是DiapatchAction的子类,也可以定义多个方法,但主要用于一个表单里有多个按钮,而这些按钮又有一个共同的名字的场合。
SwitchAction
用于子模块之间的切换。

四.ActionForward类
Action类的excute()方法返回一个ActionForward对象,它代表了web资源的逻辑抽象,这里的web资源可以是jsp页面、Java servlet、或Action。
从excute返回ActionForward可以有两种方法。
1) 动态创建一个ActionForward实例
return new ActionForward(”Failure”,”login.jsp”,true);
2) 调用ActionMappin实例的findForward方法
这个方法先从action级别找,然后在<global-forwards />级别找
return mapping.findForward(“Failure”);

热点排行