首页 > 学院 > 开发设计 > 正文

Struts控制器组件简单介绍

2019-11-18 12:54:25
字体:
来源:转载
供稿:网友

  Struts控制器组件负责接受用户请求、更新模型,以及选择合适的视图组件返回给用户。控制器组件有助于将模型层和视图层分离,有了这种分离,就可以在同一个模型的基础上得心应手地开发多种类型的视图。Struts控制器组建主要包括:
  
  ·ActionServlet组件:充当Struts框架的中心控制器。
  
  ·RequestPRocessor组件:充当每个子应用模块的请求处理器。
  
  ·Action组件:负责处理一项具体的业务。
  
  Struts框架采用控制器组件来预处理所有的客户请求,这种集中控制方式可以满足MVC设计模式的两大需求:
  
  ·首先,控制器在用户输入数据和模型之间充当媒介 / 翻译者的角色,提供一些通用功能,如安全、登入和其他针对具体用户请求的重要服务,当系统的这些通用功能出现需求变更时,部需要修改整个应用,只需要修改局部的控制器组件即可。
  
  ·其次,由于所有的请求都经过控制器过滤,因此可以降低视图组件之间,以及视图组件和模型组件之间的相互依靠关系,提高每个组件的相对独立性。由控制器组件来决定把合适的视图组件返回给用用户,这可以减少视图组件之间直接的,错综复杂的连接关系,使应用更加灵活,便于维护。
  
  Struts框架采用ActionServlet和RequestProcessor组件进行集中控制,并采用Action组件来处理单项业务。
  
  一 控制器组件的控制机制
  
  Struts的控制器组件主要完成以下任务:
  
  ·接受用户请求
  
  ·根据用户请求,调用合适的模型组件来执行相应的业务逻辑。
  
  ·获取业务逻辑执行结果。
  
  ·根据当前状态以及业务逻辑执行结果,选择合适的视图组件返回给用户。
  
  1 Action类
  
  org.apache.struts.action.ActionServlet类是Struts框架的核心控制器组件,所有的用户请求都先有ActionServlet来处理,然后再由ActionServlet把请求转发给其他组件。Struts框架只答应在一个应用中配置一个ActionServlet类,在应用的生命周期中,仅创建ActionServlet类的一个实例,这个ActionServlet实例可以同时响应多个用户请求。
  
  (a)  Struts框架初始化过程
  
  (1)  调用initInternal()方法,初始化Struts框架内在的消息资源,如与系统日志相关的同志、警告和错误消息。
  
  (2)  调用initOther()方法,从web.xml文件中加载ActionServlet的初始化参数,如config参数。
  
  (3)  调用initServlet()方法,从web.xml文件中加载ActionServlet的URL映射信息。此外还会注册web.xml和Struts配置文件所使用的DTD文件,这些DTD文件用来验证web.xml和Struts配置文件的语法。
  
  (4)  调用initModuleConfig()方法,加载并解析子应用模块的Struts配置文件;创建ModuleConfig对象,把它存储在ServletContext中。
  
  (5)  调用initModuleMessageResources()方法,加载并初始化默认子应用模块的消息资源:创建MessageResources对象,把它存储在ServletContext中。
  
  (6)  调用initModuleDataSources()方法,加载并初始化默认子应用模块的数据源。假如在Struts配置文件中没有定义元素,就忽略这一流程。
  
  (7)  调用InitModulePlugins()方法,加载并初始化默认子应用模块的所有插件。
  
  (8)  当默认子应用模块被成功地初始化后,假如还包括其他子应用模块,将重复流程(4)~(7),分别对其他子应用模块进行初始化。
  
  (b)  ActionServlet的process()方法
  
  当ActionServlet实例接受到HTTP请求之后,在doGet()或doPost()方法都会调用process()方法来处理请求。一下是ActionServlet的process()方法的源代码:
  
  protected void process (HttpServletRequest request, HttpServletResponse response)
  
  throw IOException, ServletException {
  
  ModuleUtils.getInstance().selectModule(request, getServletContext());
  
  getRequestProcessor(getModuleConfig(request)).process(request, response);
  
  }
  
  在process()方法中,首先调用org.apache.struts.util.ModuleUtils类的selectModule()方法,这个方法选择负责处理当前请求的子应用模块,然后把与子应用模块相关的ModuleConfig和MessageResources对象存储倒request范围中,这使得框架的其余组件可以方便地从request范围中读取这些对象,从而获取应用配置信息和消息资源。
  
  process()方法的第二步操作为获得RequestProcessor类的实例,然后调用RequestProcessor类的process()方法,来完成十几的预处理请求操作。
  
  (c)  扩展ActionServlet类
  
  在Sturts 1.1 之前的版本中,ActionServlet类本身包含了很多处理请求的代码。从Struts 1.1 开始,多数功能被移到 org.apache.struts.action.RequestProcessor 类中,以便减轻ActionServlet类的控制负担。
  
  尽管新版本的Struts框架答应在应用中创建矿展ActionServlet类的子类,但是这在多数情况下没有必要,因为控制器的多数控制功能位于RequestProcessor类中。
  
  假如实际应用确实需要创建自己的ActionServlet类,则可以创建一个ActionServlet的子类,然后在web.xml文件中配置这个客户化ActionServlet类。
  
  假如覆盖了init()方法,应该确保首先调用super.init(),它保证ActionServlet的默认初始化操作被执行。除了覆盖init()方法外,事实上,还可以根据十几需要覆盖ActionServlet的任何其他方法。
  
  2 RequestProcessor类
  
  对于多应用模块的Struts应用,每个子应用模块都有各自的RequestProcessor实例。在ActionServlet的process()方法中,一旦选择了正确的子应用模块,就会调用子应用模块的RequestProcessor实例的process()方法来处理请求。在ActionServlet调用这个方法时,会把当前的request和response对象传给它。
  
  Struts框架只答应应用中存在一个ActionServlet类,但是可以存在多个客户化的RequestProcessor类,每个子应用模块都可以拥有单独的RequestProcessor类。假如想修改RequestProcessor类的一些默认功能,可以覆盖RequestProcessor基类中的相关方法。
  
  (a)  RequestProcessor类的process()方法
  
  RequestProcessor类的process()方法负责实际的预处理请求操作。
  
  RequestProcessor类的process()方法一次执行一下流程:
  
  (1) 调用processMultipart()方法。假如HTTP请求方式为POST,并且请求的contentType属性以“multipart/form-data”开头,标准的HttpServletRequest对象将被重新包装,以方便处理“multipart”类型的HTTP请求。假如请求方式为GET,或者contentType属性不是“multipart”,就直接返回原始的HttpServletRequest对象。
  
  (2) 调用processPath()方法,获得请求URI的路径,这一信息可用于选择合适的Struts Action组件。
  
  (3) 调用processLocale()方法,当ControllerConfig对象的locale属性为true,将读取用户请求中包含的Locale信息,然后把Locale实例保存在session范围内。
  
  (4) 调用processContent()方法,读取ControllerConfig对象的contentType属性,然后调用response.setContentType(contentType)方法,设置响应结果的文档类型和字符编码。
  
  (5) 调用processNoCache()方法,读取ControllerConfig对象的nocache属性,假如nocache属性为true,在响应结果中将加入特定的头参数:Pragma、Cache-Control和EXPires,防止页面被存储在客户浏览器的缓存中。
  
  (6) 调用processPreprocess()方法。该方法不执行任何操作,直接返回true。子类可以覆盖这个方法,执行客户化的预处理请求操作。
  
  (7) 调用processMapping()方法,寻找和用户请求的URI匹配的ActionMapping。假如不存在这样的ActionMapping,则向用户返回恰当的错误消息。
  
  (8) 调用processRoles()方法,先判定是否为Action配置了安全角色,假如配置了安全角色,就调用isUserRole()方法判定当前用户是否具备必需的角色;假如不具备,就结束请求处理流程,向用户返回恰当的错误消息。
  
  (9) 调用processActionForm()方法,先判定是否为ActionMapping配置了ActionForm,假如配置了ActionForm,就先从ActionForm的存在范围内寻找该ActionForm实例;假如不存在,就创建一个实例。接下来把它保存在合适的范围中,保存时使用的属性key为ActionMapping的name属性。
  
  (10) 调用processActionForm()方法。假如为ActionMapping配置了ActionForm,就先调用ActionForm的reset()方法,再把请求中的表单数据组装到ActionForm中。
  
  (11) 调用processValidate()方法,假如为ActionMapping配置了ActionForm,并且ActionMapping的validate属性为true,就调用ActionForm的validate()方法。假如validate()方法返回的ActionErrors对象中包

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表