Struts2介绍(一个大的工具库)

      Struts2介绍

     Struts2是Struts的第二代产品,以WebWork为核心,采用拦截器的机制处理用户请求,使业务逻辑控制器能与Servlet API完全脱离。Struts1采用Servlet的机制处理用户请求。

     Struts 2框架的所有类都基于接口,核心接口独立于HTTP。Struts 2配置文件中的大多数配置元素都会有默认值,有助于减少在XML文件中需要进行的配置。

Struts2框架主要由三部分组成:

    核心控制器(StrutsPrepareAndExecuteFilter)、业务控制器和用户定义的业务逻辑组件。

    (也有核心控制器使用FilterDispatcher)

Struts2框架的处理流程

第1步:客户端浏览器发送一个请求。

第2步: web服务器如Tomcat收到该请求,读取配置文件,将请求 导向Struts2的StrutsPrepareAndExecuteFilter(核心控制器), 后者根据请求决定调用合适Action。

第3步:StrutsPrepareAndExecuteFilter在调用Action之前被Struts2的拦截器拦截,拦截器自动对请求应用通用功能,如数据转换,校验等。

第4步:调用Action的execute方法,该方法根据请求的参数来执行一定的操作。

第5步:依据Action的execute方法处理结果,导向不同的URL。如在execute中验证用户,验证成功可以导向成功的页面。否则重新登录

深入理解Struts2的配置文件

Struts2框架配置文件中的包就是由多个Action、多个拦截器、多个拦截器引用构成。包的作用和java中的类包是非常类似的,它主要用于管理一组业务功能相关的action,在实际应用中,我们应该把一组业务功能相关的action 放在同一个包下。

<struts>
   <!-- Struts2的action必须放在一个指定的包空间下定义-->
   <packagename="default" extends="struts-default">
      <!-- 定义处理请求URL为login.action的Action-->
       <action name="login" class=action.LoginAction">
          <!-- 定义处理结果字符串和URL之间的映射关系-->
           <result name="success">/success.jsp</result>
           <result name="input">/login.jsp</result>
       </action>
   </package>
</struts>

2、namespace命名空间配置

     考虑到同一个Web应用中需要同名的Action,Struts2以命名空间的方式来管理Action,同一个命名空间不能有同名的Action。

     Struts2通过为包指定namespace属性来为包下面的所有Action指定共同的命名空间。

包student:没有指定namespace属性。默认的命名空间是""。

包admin:指定了命名空间/admin。则如上名为adminaction.LoginAction的Action,它处理的URL为:

http://localhost:8080/chapter10web/admin/adminLogin.action

3、include包含配置

在Struts2中可以将一个配置文件分解成多个配置文件,那么我们必须在struts.xml中包含其他配置文件。

<struts>
   <includefile="struts-default.xml"/>
   <include file="struts-student.xml"/>
   <include file="struts-admin.xml"/>
   <include file="struts-user.xml"/>
   ......
</struts>

Action类文件

       在Struts2中,Action不同于struts1.x中的Action。Struts2中Action并不需要继承任何控制器类型或实现相应接口。比如struts1.x中的Action需要继承Action或者DispatcherAction。

同时Struts2中的Action并不需要借助于象struts1中的ActionForm获取表单的数据。可以直接通过与表单元素相同名称的数据成员(setter-getter函数)获取页面表单数据。

     虽然Struts2中的Action原则上不用继承任何类。但是一般需要实现Action接口或者继承ActionSupport类,重写execute方法。如果继承ActionSupport类,我们可以在的控制器中增加更多的功能。

定义Action类的两种形式:

1、基本形式:从ActionSupport类继承
    public class LoginAction extends ActionSupport{
         private String username; 
         private String password;
         /*getter-setter代码略*/
         public void validate(){…}
         public String execute()throws Exception {….}
    }
2、普通JavaBean
    package com.bean;
    public class User {
	  private String username;
	  private String password;
            /*getter-setter代码略*/
     public String execute()throws Exception {...}
    }

Action动态处理函数

1、Action默认的execute函数
    一般客户端请求url被Struts2的拦截后,根据url指定 action名称,查找相应的action,默认调用Action类的execute函数。如:
<package name="chapter10" extends="struts-default">
		<!-- 通过Action类处理才导航的Action定义 -->
  <action name="Login" class="com.action.LoginAction">
		<result name="input">/login.jsp</result>
		<result name="success">/success.jsp</result>
  </action>
</package>

2、如果不用调用默认execute()。要求调用指定函数fun()函数。
第1种方法是修改配置文件,修改action标记的method属性值
<action name="userLogin" class="com.action.LoginAction"  method="fun">
             <result name="input">/login.jsp</result>
	   <result name="success">/success.jsp</result>
 </action>


我经过测试下面这种方法好像不支持。
第2种方法是在页面form标记action属性中指定调用处理方法名,如Action类编写如下:
public class LoginAction{
      public String fun1() throws Exception{  .....    }
       public String fun2() throws Exception{ ......   }
}
在页面form标记的action属性中指定调用处理方法名。
      <form action="/login!fun1.action" method="post">
     <form action="/login!fun2.action" method="post">

3、使用通配符映射方式
  配置文件 admin_* :定义一系列请求URL是admin_*.action模式的逻辑Action
  <action name="admin_*" 
         class="action.UserAction" method="{1}">
             <result name="input">/login.jsp</result>
        <result name="success">/success.jsp</result>
  </action>
    如上,<action name=“admin_*”>定义一系列请求URL是admin_*.action模式的逻辑Action。同时method属性值为一个表达式{1},表示它的值是name属性值中第一个*的值。
例如:用户请求URL为admin_login.action时,将调用AdminAction类的login方法;用户请求URL为admin_regist.action时,将调用到AdminAction类的regist方法。

Struts2校验框架

    输入校验几乎是任何一个系统都需要开发的功能模块,我们无法预料用户如何输入,但是必须全面考虑用户输入的各种情况,尤其需要注意那些非正常输入。Struts2提供了功能强大的输入校验机制,通过Struts2内建的输入校验器,在应用程序中无需书写任何代码,即可完成大部分的校验功能,并可以同时完成客户端和服务器端的校验。

      如果应用的输入校验规则特别,Struts2也允许通过重写validate方法来完成自定义校验,另外Struts2的开放性还允许开发者提供自定义的校验器。

      Struts2中可以通过重写validate方法来完成输入校验。如果我们重写了validate方法,则该方法会应用于此Action中的所有提供服务的业务方法。

Struts2的输入校验流程如下:

1、类型转换器负责对字符串的请求参数执行类型转换,并将这此值设置成Action的属性值。

2、在执行类型转换过程中可能出现异常,如果出现异常,将异常信息保存到ActionContext中,conversionError拦截器负责将其封装到fieldError里,然后执行第3步;如果转换过程没有异常信息,则直接进入第3步。

3、通过反射调用validateXxx()方法,其中Xxx是即将处理用户请求的处理逻辑所对应的方法名。

4、调用Action类里的validate()方法。

5、如果经过上面4步都没有出现fieldError,将调用Action里处理用户请求的处理方法;如果出现了fieldError,系统将转入input逻辑视图所指定的视图资源。

步骤:

第1步:编写一个Action类,该Action接受页面提交过来的参数

第2步:在该Action相同的目录下建一个xml文件,该文件的命名为ActionName-validation.xml,其中ActionName为该Action的类名,例如LoginValidateAction-validation.xml。然后在xml配置文件中配置需要验证的字段。

第3步:在struts.xml文件中配置Action,在Action配置中必须有input视图

   <actionname="validate"class="com.action.LoginValidateAction">

        <resultname="input">/login.jsp</result>

        <result>/index.jsp</result>

   </action>

第4步:添加一个jsp页面loginvalidate.jsp,放入一个struts标签

1、基础的Struts2输入校验规则

<validators>
	 对必填校验
	<field name="requiredValidatorField">
		<field-validator type="required">
			<message>必填内容</message>
		</field-validator>
	</field>
	 必填字符串校验
	<field name="requiredStringValidatorField">
		<field-validator type="requiredstring">
			<param name="trim">true</param>
			<message>字符串必填校验</message>
		</field-validator>
	</field>
对int类型的校验
<field name="integerValidatorField">
          <field-validator type="int">
	    <param name="min">1</param>
	    <param name="max">10</param>
	    <message key="validate.integerValidatorField" />
	</field-validator>
</field>

对日期的校验
	<field name="dateValidatorField">
		<field-validator type="date">
		<param name="min">01/01/1990</param>
		<param name="max">01/01/2000</param>
		<message key="validate.dateValidatorField" />
		</field-validator>
	</field>
对email的校验
	<field name="emailValidatorField">
		<field-validator type="email">
		<message key="validate.emailValidatorField" />
		</field-validator>
	</field>

对URL的校验
	<field name="urlValidatorField">
		<field-validator type="url">
		<message key="validate.urlValidatorField" />
		</field-validator>
	</field>
对字符串长度的校验
	<field name="stringLengthValidatorField">
		<field-validator type="stringlength">
			<param name="maxLength">4</param>
			<param name="minLength">2</param>
			<param name="trim">true</param>
	<message key="validate.stringLengthValidatorField" />
		</field-validator>
	</field>
对正则表达式的校验
	<field name="regexValidatorField">
		<field-validator type="regex">
		<param name="expression">.*\.txt</param>
		<message key="validate.regexValidatorField" />
		</field-validator>
	</field>

Struts2拦截器

     拦截器,在AOP(Aspect-Oriented Programming)中用于在某个方法或字段被访问之前,进行拦截然后在之前或之后加入某些操作。拦截是AOP的一种实现策略。

     Struts 2拦截器是动态拦截Action调用的对象。它提供了一种机制,使开发者可以定义一个特定的功能模块,这个模块可以在Action执行之前或者之后运行,也可以在一个Action执行之前阻止Action执行。同时也提供了一种可以提取Action中可重用的部分的方式。

      拦截器(Interceptor)是Struts 2的核心组成部分。很多功能都是构建在拦截器基础之上的,例如文件的上传和下载、国际化、转换器和数据校验等,Struts 2利用内建的拦截器,完成了框架内的大部分操作。

      Struts2的拦截器和Servlet过滤器类似。在执行Action的execute方法之前,Struts2会首先执行在struts.xml中引用的拦截器,在执行完所有引用的拦截器的intercept方法后,会执行Action的execute方法。

       Struts2拦截器类必须实现Interceptor接口或继承AbstractInterceptor类。

    在Struts2中称为拦截器栈InterceptorStack。拦截器栈就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序依次被调用。


步骤:

第1步:建立一个拦截器的类MyInterceptor,这里的before()和after()方法是以后拦截器会执行的方法。

第2步:我们模拟一个业务组件接口ModelInterface和一个业务组件实现类ModelImpl。

第3步:创建一个动态代理类DynamicProxy,这个类是实现InvocationHandler接口。

第4步:写个类测试一下

Web中配置方式:

第1步:创建一个拦截器的触发页面test_interceptor.jsp

第2步:定义拦截器类 MyInterceptor1.java


第3步:Struts2配置文件,拦截器的映射

第4步:通过拦截器后进入 Action

第5步:通过Action处理后的视图页面interceptorsuccess.jsp


Struts2转换器

   在B/S中,将字符串请求参数转换为相应的数据类型,应该是MVC框架提供的基本功能。Struts2也提供了类型转换功能。

    在Struts2中分两种转换,一种是局部转换,另一种是全局类型转换。具体转换的实施需要一个转换类和一个自定义类。

   1、局部类型转换

     对于int,long,double,char,float等基本类型,Struts2会自动完成类型转换,像age年龄,在输入页面是String型的,到Action后会自动转换成int型。而如果是转换成其它类类型的话,就需要自定义类型转换。这样就需要一个自定义类。要定义一个转换类,需要继承ognl.DefaultTypeConverter这个类 ,这是个类型转换的类。

步骤:

第1步: 编写转换类PointConverter.java

第2步: 编写Point类。

第3步: 编写Action类。TypeConverterAction.java
第4步:编写转换属性文件。

TypeConverterAction-conversion.properties
内容为:point=com.converter.PointConverter
自定义类、转换类、action都创建好之后,要创建一个属性文件,放置在与action在同一包。该属性文件名为:action文件名-conversion.properties。文件中的内容如下:
  point = 转换类名  即 point=com.PointConverter
     注意:
(1)  point是Action的一个属性,转换类指明所使用哪个转换类对此属性进行转换。 
(2) 有两种类型的转换器:一是局部类型转换器。仅仅对某个Action的属性起作用。属性文件名:ActionName-conversion.properties 。内容:属性名=类型转换器类,如date=com.DateConverter 。存放位置与ActionName类相同路径。二是全局类型转换器。对所有Action的特定类型的属性都会生效。属性文件名:xwork-conversion.properties 。内容如java.util.Date= com.DateConverter.存放位置为WEB-INF/classes/目录下。

第5步:编写JSP页面。

类型转换的流程:

    1、用户进行请求,根据请求名在struts.xml中寻找Action。

    2、在Action中,根据请求域中的名字去寻找对应的set方法。找到后在赋值之前会检查这个属性有没有自定义的类型转换。没有的话,按照默认进行转换;如果某个属性已经定义好了类型转换,则会去检查在Action同一目录下的 action文件名-conversion.properties文件。

    3、从文件中找到要转换的属性及其转换类。

    4、然后进入转换类中,在此类中判断转换的方向。我们是先从用户请求开始的,所以这时先进入从字符串到类的转换。返回转换后的对象。流程返回Action。

    5、将返回的对象赋值给Action中的属性,执行Action中的execute()。

 6、执行完execute()方法,根据struts.xml的配置转向页面。

    7、在jsp中显示内容时,根据页面中的属性名去调用相应的get方法,以便输出。

    8、在调用get方法之前,会检查有没有此属性的自定义类型转换。如果有,再次跳转到转换类当中。

    9、在转换类中再次判断转换方向,进入由类到字符串的转换,完成转换后返回字符串。

    10、将返回的值直接带出到要展示的页面当中去显示。

Struts2上传下载

    在Java领域中,有两个常用的文件上传项目:一个是Apache组织Jakarta的Common-FileUpload组件(http://commons.apache.org/fileupload/),另一个是Oreilly组织的COS框架(http://www.servlets.com/cos/)。利用这两个框架都能很方便的实现文件的上传。

      上传文件主要是通过读写二进制流进行操作的。Form表单元素的enctype属性指定的是表单数据的编码方式。  

      multipart/form-data。这种编码方式的表单会以二进制流的方式来处理表单数据,这种编码方式会把文件域指定文件的内容也封装到请求参数里。

上传:

struts.xml

<action name="fileUpLoad" class="cn.hncu.user.fileupload.FileUpLoadAction">
			<!-- 动态设置Action中的savePath属性的值 --> 
			<param name="savePath">/upload</param>
			<result name="input">/jsps/uploadfile.jsp</result>
			<result name="success">/jsps/success.jsp</result>
		</action>


接收文件的类:
package cn.hncu.user.fileupload;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;

import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionSupport;

public class FileUpLoadAction extends ActionSupport{
	private static final int BUFFER_SIZE=20*1024;
	private String title;//文件标题
	private File upload; //上传文件对象==传文件的内容所对应的属性名,本例取:upload--前端file组件的参数名
	private String uploadFileName; //上传文件名=upload+"FileName"
	private String uploadContentType;//上传文件类型=  upload+"ContentType"

	
	private String savePath;//保存文件的目录路径(通过依赖注入)
	
	private static void copy(File src,File dest){
		InputStream in=null;
		OutputStream out=null;
		
		try {
			in =new BufferedInputStream(new FileInputStream(src),BUFFER_SIZE);
			out=new BufferedOutputStream(new FileOutputStream(dest), BUFFER_SIZE);
			
			byte[] buffer=new byte[BUFFER_SIZE];
			int len=0;
			while((len=in.read(buffer))>0){
				out.write(buffer,0,len);
			}
			in.close();
			out.close();
		} catch (FileNotFoundException e) {
			e.printStackTrace();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
	public String getTitle() {
		return title;
	}

	public void setTitle(String title) {
		this.title = title;
	}


	public File getUpload() {
		return upload;
	}

	public void setUpload(File upload) {
		this.upload = upload;
	}

	public String getUploadFileName() {
		return uploadFileName;
	}

	public void setUploadFileName(String uploadFileName) {
		this.uploadFileName = uploadFileName;
	}

	public String getUploadContentType() {
		return uploadContentType;
	}

	public void setUploadContentType(String uploadContentType) {
		this.uploadContentType = uploadContentType;
	}

	public String getSavePath() {
		return savePath;
	}

	public void setSavePath(String savePath) {
		this.savePath = savePath;
	}
	@Override
	public String execute() throws Exception {
		//根据服务器的文件保存地址和源文件名创建目录文件全路径
		String destPath=ServletActionContext.getServletContext().getRealPath(
				this.getSavePath())+"/"+this.getUploadFileName();
		ServletActionContext.getServletContext().getRealPath(this.getSavePath());
		System.out.println(this.getSavePath());
		System.out.println("上传的文件的类型:"+this.getUploadContentType());
		System.out.println("destPath  :....."+destPath);
		File destFile=new File(destPath);
		copy(this.upload,destFile);
		return "success";
	}
	
}

下载:

<action name="down" class="cn.hncu.user.fileupload.FileDownAction">
			<!-- 
			 <result name="success" >/index.jsp
			</result>
			 -->
			<result name ="success" type="stream">
				<param name="contentType">
					application/vnd.ms-word
				</param>
				<param name="contentDisposition">
				   attachment;filename="a.docx"
				</param>
				<param name="inputName">downloadFile</param>
			</result> 
		</action>

package cn.hncu.user.fileupload;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.InputStream;

import com.opensymphony.xwork2.ActionSupport;

public class FileDownAction extends ActionSupport{
	private InputStream downloadFile;
	public InputStream getDownloadFile(){
		File file=new File("d:/a/a.docx");
		try {
			downloadFile=new FileInputStream(file);
		} catch (FileNotFoundException e) {
		}
		return downloadFile;
	}
	@Override
	public String execute() throws Exception {
		System.out.println("进来下载文件了......");
		return "success";
	}
	
}


Struts2标签使用

这里给出了大部分常用的标签

<%@ page language="java" pageEncoding="gb2312"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<!-- struts2标签库调用声明 -->
<%@taglib prefix="s" uri="/struts-tags"%>
<html>
	<head>
	</head>
	<body>
		<s:form action="aa" method="post">
			<b>Textfield标签--单行文本输入控件</b><br/>
			<s:textfield label="用户名" name="uname" value="admin" size="20"></s:textfield><br/>
			
			<b>Textarea标签--多行文本输入控件</b><br/>
			<s:textarea name="note" label="备注" rows="4" cols="10"></s:textarea><br/>
			
			<b>Radio标签--单选按钮</b><br/>
			<s:radio label="性别" name="sex" list="#{'male':'男','female':'女'}"></s:radio>
			
			<b>checkboxlist--标签复选框</b>
			<s:checkboxlist name="hobby" label="爱好" list="{'体育','音乐','写代码'}"></s:checkboxlist>
			
			<b>Select标签--下拉列表框</b>
			<i>使用name和list属性,list属性的值是一个列表:</i>
			<s:select label="最终学历" name="education" multiple="true" list="{'小学','中学','本科','硕士','博士','其他'}" size="3"></s:select>
			
			<b>doubleselect标签--关联HTML列表框,产生联动效果</b>
			<s:doubleselect label="请选择居住地" name="province" doubleList="top=='安徽省'?{'合肥市','马鞍山市','芜湖市'}:
                  {'杨浦区','浦东新区','静安区','闸北区'}" list="{'安徽省','上海市'}" doubleName="city"></s:doubleselect>
                  
             <b>updownselect标签--带有上下移动的按钮的列表框</b>
             <s:updownselect name="books" label="请选择您喜欢的书籍" labelposition="top"
              moveUpLabel="up" moveDownLabel="down" selectAllLabel="all"
             list="#{'1':'数据结构原理','2':'数据库原理','3':'Java编程','4':'C语言基础' }"
             listKey="key" listValue="value" size="2"
             ></s:updownselect>
           	
           	<b>optiontransferselect标签</b>
           	<s:optiontransferselect label="最喜欢的节日" name="lakedays" doubleList="{'元旦', '圣诞节', '万圣节'}" 
           	list="{'五一劳动节', '国庆节', '春节'}" doubleName="cBook"></s:optiontransferselect>
		</s:form>
	</body>
</html>




<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="/struts-tags" prefix="s"%><!-- Struts2标签库调用声明 -->

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
  </head>
  <%List<Object> lists=new ArrayList<Object>();
  	lists.add("aa");
  	lists.add("bb");
  	lists.add("cc");
  	lists.add("dd");
  	lists.add("ee");
  	request.setAttribute("lists", lists);
   %>
  <body>
  	<s:iterator value="#request.lists" var="list" status="stuts">
  		<s:if test="#stuts.odd==true">
  			<li><font color="red">${list}</font></li>
  		</s:if>
  		<s:else>
  			<li><s:property/> </li>
  		</s:else>
  	</s:iterator>
  </body>
</html>









  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值