struts1.2原理

来源:互联网 发布:php和java的关系 编辑:程序博客网 时间:2024/05/20 14:44

status学习笔记及历程

第一讲:

1. 了解struts工作流程:

首先struts基于MVC模式, 用户发送的请求讲被ActionServlet处理,转发,但是它是怎样实现的了??

实现原理:
web.xml 配置:

<servlet>

<servlet-name>action</servlet-name>

<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>

<init-param>

<param-name>config</param-name>

<param-value>/WEB-INF/struts-config.xml</param-value>

</init-param>

... ...

</servlet>

<servlet-mapping>

<servlet-name>action</servlet-name>

<url-pattern>*.do</url-pattern>

</servlet-mapping>上面描述*.do的url请求都会被ActionServlet接受.

完整配置讲解: 1. web.xml

web.xml文件对任何的Web项目都是一个必须的文件,使用Struts时,还需要对该文件进行一
些必须的配置。

1.1 ActionServlet的配置一般需要在该文件中配置Struts的Servlet,示例配置如下:

Eg1. 简单的Struts的ActionServlet的配置:

<servlet> <servlet-name>action</servlet-name>

<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>

<init-param>

<param-name>config</param-name>

<param-value>/WEB-INF/struts-config.xml</param-value>

</init-param>

<init-param>

<param-name>debug</param-name>

<param-value>3</param-value>

</init-param>

<init-param>

<param-name>detail</param-name>

<param-value>3</param-value>

</init-param>

<load-on-startup>0</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>action</servlet-name>

<url-pattern>*.do</url-pattern>

</servlet-mapping>

对于复杂的应用,一般需要配置多个struts-config.xml文件,可以通过添加另外的
<init-param>来实现,或者在多个配置文件中以为“,”隔开,如下所示:

Eg2. 配置多个struts-config.xml配置文件的ActionServlet的配置:

<servlet>

<servlet-name>action</servlet-name>

<servlet-class>

org.apache.struts.action.ActionServlet

</servlet-class>

<init-param>

<param-name>config</param-name>


<param-value>/WEB-INF/struts-config.xml</param-value
>

</init-param>

<init-param>

<param-name>config/IVR</param-name>


<param-value>/WEB-INF/struts-config-IVR.xml</param-v
alue>

</init-param>

<init-param>

<param-name>config/wap</param-name>

<param-value>

/WEB-INF/struts-config-wap.xml

</param-value>

</init-param>

<init-param>

<param-name>debug</param-name>

<param-value>3</param-value>

</init-param>

<init-param>

<param-name>detail</param-name>

<param-value>3</param-value>

</init-param>

<load-on-startup>0</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>action</servlet-name>

<url-pattern>*.do</url-pattern>

</servlet-mapping>

1.2 欢迎和错误处理的配置
首先讲述一下欢迎文件清单<welcome-file-list>的配置,该元素可包含多个<welcome-file
>子元素,当Web容器调用欢迎界面时,将首先查看第一个<welcome-file>子元素中定义的文
件是否存在,若存在,则将其返回给用户,若不存在,继续判断第二个<welcome-file>子元
素中定义的文件……,配置示例如下:

<welcome-file-list>

<welcome-file>index.html</welcome-file>

<welcome-file>index.jsp</welcome-file>

<welcome-file>default.jsp</welcome-file>

</welcome-file-list>

接着讲述一下在web.xml中如何配置错误处理,这时需要使用<error-page>元素,该置示例如下:

<!-- 根据错误码进行跳转-->

<error-page>

<error-code>500</error-code>

<location>/error.jsp</location>

</error-page>

<!-- 根据异常进行跳转-->

<error-page>

<exception-type>java.lang.NullException</exception-type>

<location>/error.jsp</location>

</error-page>

1.3 tld文件的配置
若Web工程没有使用Struts的标签库,可以不在web.xml中使用Struts的标签库信息。当然若
开发人员使用了struts的标签库,也可以直接在jsp页面中引入标签库,例如通过如下方式引
入:

<%@ taglib uri="/WEB-INF/struts-bean.tld" prefix="bean"%>

<%@ taglib uri="/WEB-INF/struts-html.tld" prefix="html"%>

<%@ taglib uri="/WEB-INF/struts-logic.tld" prefix="logic"%>

<%@ taglib uri="/WEB-INF/struts-nested.tld" prefix="nested"%>


在Struts中进行配置的的好处是因为可以在Struts中配置为tld文件配置一个简要的
名称或者更加易懂的名称,例如在web.xml文件中增加如下配置:

<taglib>

<taglib-uri>/tags/struts-bean</taglib-uri>

<taglib-location>/WEB-INF/struts-bean.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-html</taglib-uri>

<taglib-location>/WEB-INF/struts-html.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-logic</taglib-uri>

<taglib-location>/WEB-INF/struts-logic.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-nested</taglib-uri>

<taglib-location>/WEB-INF/struts-nested.tld</taglib-location>

</taglib>


其中<taglib-uri>元素指定标签库的相对或者绝对URI地址,Web应用将根据这一URI
来访问标签库;<taglib-location>元素指定标签库描述文件在文件资源系统中的物
理位置。

此时在jsp页面通过如下方面引入标签库:

<%@ taglib uri="/tags/struts-bean " prefix="bean"%>

<%@ taglib uri="/tags/struts-html" prefix="html"%>

<%@ taglib uri="/tags/struts-logic " prefix="logic"%>

<%@ taglib uri="/tags/struts-nested " prefix="nested"%>

1.4 完整配置实例
下面举一个使用Struts的Web项目的web.xml的简单配置实例(该实例开发人员也参考struts
-1.2.8-bin.zip包的webapps目录下的struts-mailreader.war),内容如下所示:

<?xml version="1.0" encoding="ISO-8859-1"?>

<!DOCTYPE web-app

PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.2//EN"

"http://java.sun.com/j2ee/dtds/web-app_2_2.dtd">

<web-app>

<display-name>Struts Example Application</display-name>

<!-- Action Servlet Configuration -->

<servlet>

<servlet-name>action</servlet-name>

<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>

<init-param>

<param-name>config</param-name>

<param-value>/WEB-INF/struts-config.xml,
/WEB-INF/struts-config-registration.xml</param-value>

</init-param>

<load-on-startup>1</load-on-startup>

</servlet>

<!-- Action Servlet Mapping -->

<servlet-mapping>

<servlet-name>action</servlet-name>

<url-pattern>*.do</url-pattern>

</servlet-mapping>

<!-- 欢迎列表-->

<welcome-file-list>

<welcome-file>index.jsp</welcome-file>

</welcome-file-list>

<!-- 错误处理 -->

<error-page>

<exception-type>java.lang.Exception</exception-type>

<location>/error.jsp</location>

</error-page>

<taglib>

<taglib-uri>/tags/struts-bean</taglib-uri>

<taglib-location>/WEB-INF/struts-bean.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-html</taglib-uri>

<taglib-location>/WEB-INF/struts-html.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-logic</taglib-uri>

<taglib-location>/WEB-INF/struts-logic.tld</taglib-location>

</taglib>

<taglib>

<taglib-uri>/tags/struts-nested</taglib-uri>

<taglib-location>/WEB-INF/struts-nested.tld</taglib-location>

</taglib>

</web-app>

2 .properties资源文件
默认情况下,Struts默认的资源文件为ApplicationResources.properties文件。

资源文件可为一个束,可针对不同的语言,写不同的资源文件,开发人员可以定义英文、简
体中文、繁体中文、日文等资源文件,资源文件的命名格式为:资源文件名_国别_语言.pro
perties,常见的资源文件名称为:

ApplicationResources.properties:默认资源文件,或英文资源文件

ApplicationResources_zh_CN.properties:简体中文

ApplicationResources_zh_TW.properties:繁体中文


每一个资源文件都是“键-值”对的集合,例如可在src目录下的资源文件Application
Resources.properties中,编写如下内容:

UserForm.userName = USERNAME

UserForm.password = PASSWORD


同时在src目录下建立ApplicationResources_zh_CN.properties.bak文件来编写简体
中文的原始资源文件,内容如下:

UserForm.userName = 用户名

UserForm.password = 密码


开发人员还可以建立ApplicationResources_zh_TW.properties.bak文件编写繁体中
文的原始资源文件。其内容如下:

UserForm.userName = ノめ?W

UserForm.password = ノめ?W


对于简体中文和繁体中文文件,开发人员还需要对其使用native2ascii工具对其进行
转换,将非ASCII码转换为Unicode编码,native2ascii工具在%JAVA_HOME%/bin目录
下的native2ascii.exe,因此若要进行这种转换,读者首先需要安装了该工具。安装
了该工具之后,进入其所在目录,运行native2ascii…命令可进行编码转换,为了能
在命令行直接使用该命令,读者还需要在系统变量PATH中添加路径:%JAVA_HOME%"b
in。

在实际项目中,一般编写一个批处理文件(eg.
code.bat)来处理资源文件的编码,例如code.bat为如下内容时,可满足要求将如上
的ApplicationResources_zh_CN.properties.bak文件进行编码,并将编码后的信息
放入ApplicationResources_zh_CN.properties文件中。该批处理文件的内容如下所
示:

del ApplicationResources_zh_CN.properties

copy ApplicationResources_zh_CN.properties.bak
ApplicationResources_zh_CN.properties.gbk

native2ascii -encoding GBK ApplicationResources_zh_CN.properties.gbk
ApplicationResources_zh_CN.properties

del ApplicationResources_zh_CN.properties.gbk

del ApplicationResources_zh_TW.properties

copy ApplicationResources_zh_TW.properties.bak
ApplicationResources_zh_TW.properties.big5

native2ascii -encoding Big5 ApplicationResources_zh_TW.properties.big5
ApplicationResources_zh_TW.properties

del ApplicationResources_zh_TW.properties.big5

del *.bak.bak


运行code.bat之后,可看到目录下多出了两个资源文件,即:ApplicationResource
s_zh_CN.properties和ApplicationResources_zh_TW.properties。其中Applicatio
nResources_zh_CN.properties的内容是经过编码的,内容如下:

UserForm.userName = "u7528"u6237"u540d

UserForm.password = "u5bc6"u7801


在jsp页面中,可以通过Struts的自定义标签来获取资源文件的某个键(例如:User
Form.userName)的信息。示例如下:

<bean:message key="UserForm.userName"/>

**************************************************************/

1.检索和用户请求匹配的Actionmapping(文件配置在struts-config.xml文件中),对于一个表单在文件中需要实现如下配置

<!-- 配置表单实体 -->
<form-beans>
<form-bean name="loginForm" type="com.youcompany.forms.LoginForm" />
</form-beans> name="loginFrom" 对应JSP页面表单的name属性
type="com.youcompnay.forms.LoginForm" 对应该表单的处理类,该类集成struts的ActionForm可作验证,类型转换等功能,
负责将form包装成一个form对象

<!-- 表单处理及后期转换-->
<action-mappings>
<action path="/login" type="com.yourcompany.actions.LoginAction"
name="loginForm" scope="request" input="/login.jsp" >
<forward name="success" path="/menu.jsp"></forward>
<forward name="fails" path="/login.jsp"></forward>
</action>
</action-mappings>

属性介绍
<action>部分是说明action的属性。Type 指定Action的类名,负责处理此次form表单请求的类
Name 指定Action主力的ActionForm名,与<form-beans >元素的name属性匹配。
Scope 指定ActionForm存在的范围, (request, application)
Input 指定包含客户提交表单的网页,如果ActionForm的Validate方法返回错误,则因该把用户Validate 如果取值为true,则表示ActionServlet应该调用ActionForm的validate方法
Forward 就是Action的execute方法执行完毕后,把客户请求在转发给相应的页面。

总结,对于struts每一个form表单处理, 常见操作是建立如下的package com.yourcompany.forms
com.youcompany.action
struts-config.xml
配置文件,ActionServlet将检查该配置,并实现流程串接.

2. 如果ActionForm实例不存在, 就创建一个ActionForm对象,把客户提交的表单保存到ActionForm对象中

总结: 自动创建的ActionForm主要实现表单字段的getter,setter方法, 如果表单字段需要转换成各种对象,其中涉及到类型转换的知识:

struts的类型转换:??????

3. 根据配置信息决定是否需要进行表单验证, 如果需要验证, 就调用ActionForm的validate()方法 ActionForm的Validate()方法, 这个很简单,你只需要在ActionForm的包装类中重写validate()方法即可,如果验证不成功,请返回

4. 如果ActionForm的validate()方法返回null或返回一个不包含ActionMessage的ActionErrors对象,就表示表单验证成功.

5. ActionServlet根据ActionMapping实例包含的映射信息决定将请求转发给哪个Action, 如果Action实例不存在,就先创建这个实例,
然后调用Action的execute方法

6. Action的execute返回ActionForward对象,ActionServlet在把客户请求转发到ActionForward对象指向到JSP组件.

7. ActionForward对象指向的JSP组件动态生成网页,返回给客户.

第二讲:

helloapp演示例子中掌握:

1. 分析应用需求
2. 实际操作MVC的struts框架.
3. 视图的创建
4. 资源文件application.properties资源文件的配置和作用
5. 数据合法性验证,表单事务逻辑处理
6. 控制器组件HelloAction.java(处理表单)
7. 模型组件: PersonBean.java 实体包装类
8. 模块贡献的java文件: Constants.java
9. web.xml, struts-config.xml配置
10. 编译,发布

1. 分析应用需求
通常就是你需要实现的业务功能,若注册,登录,订单

2. 实际操作MVC的struts框架.

MVC构成:

M(model) 包含PersonBean的JavaBean组件,提供属性的getter,setter方法,及其他方法
V(view) JSP, 提供界面设置,可直接用html标签,也可以用struts的tag-lib
C(controller) HelloAction, 主要做: 1.业务逻辑验证, 2.调用模型组件 3.返回视图组件

配置文件:
struts-config.xml


3. 视图的创建
Hello.jsp 代码:
<%@ page language="java" import="java.util.*" pageEncoding="ISO-8859-1"%>
<%@ page taglib uri="/WEB-INF/struts-bean.tld" prefix="bean" %>
<%@ page taglib uri="/WEB-INF/struts-html.tld" prefix="html" %>
<%@ page taglib uri="/WEB-INF/struts-login.tld" prefix="login" %>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>

<html:html locale="true">
<head>
<title><bean:message key="hello.jsp.title" /></title>
<html:base />
</head>

<body bgcolor="white">
<h2><bean:message key="hello.jsp.page.heading" /></h2>
<p>
<html:errors/>
</p> </login:present>
<html:form action="/HelloWorld.do" focus="userName">
<bean:message key="hello.jsp.prompt.person"/>
<html:test property="userName" size="16" maxlength="16" /><br/>
<html:submit property="submit" value="submit"/>
<html:reset/>
</html:form> </html:form><br/>
<html:img page="http://www.huomo.cn/struts-power.gif" alt="Powered by Struts"/> </body>
</html:html>

解释:

struts标签库的加载
<%@ page taglib uri="/WEB-INF/struts-bean.tld" prefix="bean" %>
<%@ page taglib uri="/WEB-INF/struts-html.tld" prefix="html" %>
<%@ page taglib uri="/WEB-INF/struts-login.tld" prefix="login" %>

web.xml 定义标签库代码:
<taglib>
<taglib-uri>/WEB-INF/nested.tld</taglib-uri>
<taglib-location>/WEB-INF/nested.tld</taglib-location>
</taglib><taglib>
<taglib-uri>/WEB-INF/struts-bean.tld</taglib-uri>
<tadlib-uri>/WEB-INF/struts-bean.tld</tadlib-uri>
</taglib><taglib>
<taglib-url>/WEB-INF/struts-html.tld</taglib-url>
<taglib-location>/WEB-INF/struts-heml.tld</taglib-location>
</taglib><taglib>
<taglib-uri>/WEB-INF/struts-logic.tld</taglib-uri>
<taglib-location>/WEB-INF/struts-logic.tld</taglib-location>
</taglib>

常见JSP标签解析:
<html:errors>: struts框架产生的错误信息
<html:form>
<html:text>
<bean:message>
<bean:write>

更多标签......

4. 资源文件application.properties资源文件的配置和作用
Hello.jsp页面中使用了<bean:message>标签,这些key对应的就是application.properties里面配置好的信息段, 这些配置对实现
国际化有突出意义

hello.jsp.title=Hello~A first struts program
hello.jsp.page.heading=Hello World!A first Status application
..............

5. 数据合法性验证,表单事务逻辑处理
LoginForm.java

public ActionErrors validate(ActionMapping mapping, HttpServletRequest request){
ActionErrors errors = new ActionErrors();
if( (username == null) || (username.length() < 1)){
errors.add("usrename", new ActionMessage("hello.no.username.error"));
return errors;return null;
  }

6. 控制器组件HelloAction.java(处理表单)
public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request,
HttpServletResponse response) throws IOException, ServletException

所有的表单Action通过验证之后,都会调用execute()方法,
ActionMapping: 包含Action的配置信息和struts-config.xml文件中的action元素
ActionForm: 包含了用户表单数据信息
HttpServletRequest: 当前http请求对象
HttpServletResponse: 当前HTTP相应的对象

7. 模型组件: PersonBean.java 实体包装类 package com.imc.forms;

import javax.servlet.http.HttpServletRequest;

import org.apache.struts.action.ActionErrors;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.action.ActionMessage;

public class LoginForm extends ActionForm{
private static final long serialVersionUID = 3348342323123L;private String username;
private String password;
public String getUsername() {
return username;public void setUsername(String username) {
this.username = username;public String getPassword() {
return password;public void setPassword(String password) {
this.password = password;
}public void reset(ActionMapping mapping, HttpServletRequest request){
this.password = password;
this.username = null; //
public ActionErrors validate(ActionMapping mapping, HttpServletRequest request){
ActionErrors errors = new ActionErrors();
if( (username == null) || (username.length() < 1)){
errors.add("usrename", new ActionMessage("hello.no.username.error"));
return errors; return null;}

8. 模块贡献的java文件: Constants.java

public final class Constants{
public static final String PERSON_KEY = "personbean";
}

9. web.xml, struts-config.xml配置 <message-resource parameter="hello.application" /> WEB-INF/classes/hello/application.properties

10. 编译,发布编辑打包war..

总结: 服务器端执行表单验证流程如下:

1. servlet容器在web.xml文件中寻找<url-pattern>属性为*.do的<servlet-mapping>元素
<servlet-mapping>
<servlet-name>action</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>

2. servlet容器依据以上<servlet-mapping>元素的<servlet-name>属性action,
在web.xml文件中寻找匹配的<servlet>元素
<servlet>
<servlet-name>action</servlet-name>
<servlet-class>org.apache.struts.action.ActionServlet</servlet-class>
</servlet>

3. servlet容器把请求转发到<servlet-class>, ActionServlet依据用户请求路径HelloWorld.do, 在struts配置文件中检索
path属性为/HelloWorld的action元素

<action path="/HelloWorld"
type = "hello.HelloAction"
name = "HelloForm"
scope = "request"
validate = true
input = "/hello.jsp"

4 ActionServlet根据,action元素的name属性, 创建一个HelloForm对象, 把客户提交的表单数据传给HelloForm对象,
再把HelloForm对象保存再scope指定的request返回内

5 由于Action元素的validate属性为true, ActionServlet调用HelloForm对象的validate进行表单验证,事实上validate总是为true,
只要你重写了public ActionErrors validate(ActionMapping mapping, HttpServerRequest request) 都会验证表单

 

原创粉丝点击