struts学习笔记

Struts学习笔记
注:


******************************************************************************************************************

23:41 2012/3/18
Struts2优点
1、在软件设计上struts2没有像struts1那样跟servletAPI和strutsAPI有着紧密的耦合,struts2的应用可以不依赖与servletPAI和strutsAPI。struts2的这种设计上与无侵入式设计,而struts1却属于侵入式设计。
public class OrderListAction extends Action{
    public ActionForward execute(ActionMapping mapping,ActionForm form,
                HttpServletRequest request,HttpServletResponse response)
                throws Exception{
    }
}
2、struts2提供了拦截器,利用拦截器可以进行AOP编程,实现如权限拦截等功能。
3、struts提供了类型转换器,我们可以把特殊的请求参数转换成需要的类型,在struts1中,如果我们需要实现同样的功能,就必须向struts1的底层实现BeanUtil注册类型转换器才行。
4、struts2提供支持多种表现层技术,如:jsp、freeMarker、Velocity等。
5、struts2的输入校验可以对指定方法进行校验,解决了struts1长久之痛。
6、提供了全局范围、包范围和Action范围的国际化资源文件管理实现。

******************************************************************************************************************

23:54 2012/3/18
struts2开发环境配置
1、struts2所需要用到的jar包文件
2、编写struts2的配置文件
3、在web.xml中加入struts2 MVC框架启动配置
******************************************************************************************************************

1:12 2012/3/20
搭建struts2开发环境——开发struts2应用依赖的jar文件
大家可以到http://struts.apache.org/download.cgi#struts2014下载struts2-2.x.x-all.zip,目前最新版本为2.1.6,下载完之后解压文件,开发struts2应用需要依赖的jar文件在解压目录的lib文件夹下,不同的应用需要的jar包是不同的。下面给出了开发struts程序最少需要的jar。
struts2-core-2.x.x.jar:struts2框架的核心类库
xwork-2.x.x.jar:xwork类库,struts2在其上构建
ognl-2.6.x.jar:对象图导航语言(Object Graph Navigation language),struts2框架通过其写对象的属性
freemarker-2.3.x.jar:struts2的UI标签的模板使用FreeMarker编写
commons-logging-1.1.x.jar:ASF出品的日志包,struts2框架使用这个日志包来支持log4j和JDK1.4+的日志记录
commons-fileupload-1.2.1.jar文件上传组件,2.1.6版本后必须加入此文件
******************************************************************************************************************

1:26 2012/3/20
搭建struts2开发环境——struts2应用的配置文件
struts2默认的配置文件为struts.xml,该文件需要存放在WEB-INF/classes下,该文件的配置模板如下:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
    "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
    "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>
</struts>

******************************************************************************************************************

1:41 2012/3/20
搭建struts2开发环境——struts2在web中的启动配置
在struts1.x中,struts框架是通过servlet启动的。在struts2中,struts框架式通过Filter启动的。他在web.xml中的配置如下:
  <filter>
        <filter-name>struts2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
    </filter>

    <filter-mapping>
        <filter-name>struts2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
在StrutsPrepareAndExcuteFilter的init()方法中将会读取类路径下默认的配置文件struts.xml完成初始化操作。
注意:struts2读取到struts.xml的内容后,以javabean形式存放在内存中,以后struts2对用户的每次请求处理将使用内存中的数据,而不是每次都读取struts.xml文件

******************************************************************************************************************

22:54 2012/3/20
struts.xml配置中的包介绍
<package name="itcast" namespace="/test" extends="struts-default">
    <action name="helloworld" class="cn.itcast.action.HellWorldAction" method="execute">
        <result name="success">/WEB-INF/page/hello.jsp</result>
    </action>
</package>
1、在struts2框架中使用包来管理action,包的作用和java中的类包是非常类似的,它主要用于管理一组业务功能相关的action。在实际应用中,我们应该把一组业务功能相关的action放在同一个包下。
2、配置包是必须制定name属性,该name属性值可以任意取名,但必须唯一,他不对应java的类包,如果其他包要继承该包,必须通过该属性进行引用。包的namespace属性用于定义该包的命名空间,命名空间作为访问该包下action的路径一部分,如访问上面例子的action,访问路径为:/test/helloworld。action。namespace属性可以不配置,对本例而言,如果不指定该属性,默认的命名空间为""(空字符串)。
3、通常每个包都应该继承struts-default包,因为是struts2很多核心的功能都是拦截器来实现。如:从请求中把请求参数封装到action、文件上传和数据验证等等都是通过拦截器实现的。struts-default定义了这些拦截器和Result类型。可以这么说:当包继承了struts-default才能使用struts2提供的核心功能。struts-default包是在struts2-core-2.x.x.jar文件中的struts-default.xml中定义。struts-default.xml也是struts2默认配置文件。struts2每次都会自动加载struts-default.xml文件。
4、包还可以通过abstract="true"定义为抽象包,抽象包中不能包含action。
******************************************************************************************************************

0:28 2012/3/21
Action名称的搜索顺序
1、获得请求路径的url,例如url是:http://server/struts2/paht1/path2/path3/test.action
2、首先寻找namespace为/paht1/path2/path3的package,如果不存在这个package则执行步骤3;如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action是就会直接跑到默认namespace的package里面去寻找action(默认的命名空间为空字符串""),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action。
3、寻找namespace为/paht1/path2的package,如果不存在这个package,则转至步骤4;如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action是就会直接跑到默认namespace的package里面去寻找action(默认的命名空间为空字符串""),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action。
4、寻找namespace为/paht1的package,如果不存在这个package,则转至步骤4;如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action是就会直接跑到默认namespace的package里面去寻找action(默认的命名空间为空字符串""),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action。
5、寻找namespace为/的package,如果不存在这个package,则转至步骤4;如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action是就会直接跑到默认namespace的package里面去寻找action(默认的命名空间为空字符串""),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action。

******************************************************************************************************************

0:55 2012/3/21
Action配置中的各项默认值
1、如果没有为action指定class,默认是ActionSupport。
2、如果没有为action指定method,默认执行action中的execute()方法。
3、如果没有指定result的name属性,默认值为success。
******************************************************************************************************************

12:54 2012/3/21
result各种视图转发
<result>/index.jsp</result>
<resutl name="add" >/index.jsp</result>
<result name="add" redirect="true">/index.jsp</result>
1、以上第一种为默认转发
2、第二种为根据返回值转发
3、第三种根据返回值重定向


result配置类似于struts1中的forward,但struts中提供了各种结果类型,常用的类型有:dispatcher(默认值)、redirect、redirectAction、plainText。
在result中还可以使用${}表达式访问action中的属性,表达式里的属性名称对应action中的属性。
注:转发路径不能再WEB-INF文件里面
plaintext显示原始文件内容,例如我们需要原样显示jsp文件源代码的时候,我们可以使用此类型。
<result>
    <param name="location">/xxxx.jsp</param>
    <param name="charSet">UTF-8</param>
</result>
******************************************************************************************************************

17:18 2012/3/21
为action的属性值入值
struts2为action中的属性提供了依赖注入功能,在struts的配置文件中,我们可以很方便地为action中的属性注入值,注意:属性必须提供setter方法
public class HelloWroldAction{
    private String savePath;
    public String getSavePath(){
        return savePath;
    }
    public void setSavePath(String savePath){
        this.savePath = savePath;
    }
    ……
}
<package name="flame" namespace="/test" extends="struts-default">
    <action name="helloworld" class="cn.flame.HelloWorldAction">
            <param name="savePath">/image</param>
            <result name="success" >/WEB-INF/page/hello.jsp</result>
    </action>
</package>
上面通过<param>节点为action的savePath属性注入“/image”
******************************************************************************************************************

0:04 2012/3/22
常用常量
<!-- 指定默认编码集,作用于HttpServletRequest的setCharacterEncoding方法和freemarker、velocity的输出 -->
<constantname="struts.i18n.encoding" value="UTF-8"/>
<!--属性指定需要struts2处理的请求后缀,该属性的默认值是action,即所有匹配*。action的请求都由struts2处理。
如果用户需要指定多个请求后缀,则多个后缀之间一英文逗号隔开 -->
<constant name="struts.action.extension" value="do" />
设置浏览器是否缓存静态内容,默认值为true(生产环境下使用)、开发阶段最好关闭
<constant name="struts.serve.static.browserCache" value="false" />
当struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false(生产环境下使用),开发阶段最好打开
<constant name="struts.configuration.xmlreload" value="false" />
开发模式下使用,这样可以打印出更详细的错误信息
<constant name="struts.devMode" value="true" />
默认的视图主题
<constant name="struts.ui.theme" value="simple" />
与spring集成是,指定由spring负责action对象的创建
<constant name="struts.objectFactory" value="spring" />
该属性设置strut2是否支持动态方法调用,该属性的默认值为true,如果需要关闭动态方法调用,则可设置该属性为false
<constant name="struts.enable.DynamicMethodInvocation" value="false" />
上传文件大小限制
<constant name="struts.multipart.maxSize" value="10701096" />
******************************************************************************************************************
15:44 2012/3/23
struts处理流程
用户请求    →    StrutsPrepareAndExecuteFilter    →    Interceptor(struts内置的一些拦截器或用户自定义拦截器)    →    Action(用户编写的action类,类似struts1中的action)    →    Result(类似struts1中的forward)    →    JSP/HTML

StrutsPrepareAndExecuteFilter是struts2框架的核心控制器,它负责拦截由<url-pattem>/*</url-pattem>指定的所有用户请求,当用户请求到达时,该Filter会过滤用户的请求。默认情况下,如果用户请求的路劲不带后缀或者后缀以action结尾,这时请求将被拦转入struts2框架处理,否则struts2框架将略过该请求的处理。当请求转入struts2框架处理时会先经过一系列的拦截器,然后再到action。与struts1不同,struts2对用户的每一次请求都会创建一个action,所以struts2中的action是线程安全的。
******************************************************************************************************************

16:01 2012/3/23
为应用指定多个struts配置文件
在大部分应用里,随着应用规模的增加,系统中的action的数量也会大量增加,导致struts.xml配置文件变得非常臃肿,为避免struts.xml文件过于庞大、臃肿,提高struts.xml文件的可读性,我们可以将一个struts.xml配置文件分解成多个配置文件,然后在struts.xml文件中包含其他配置文件。
下面的struts.xml通过<include>元素指定多个配置文件:
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
    "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
    "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>
    <!-- Add packages here -->
    <include file="struts-users.xml"/>
    <include file="struts-order.xml"/>
</struts>


通过这种方式,我们就可以将struts的action按模块添加在多个配置文件中。
******************************************************************************************************************

16:56 2012/3/23
动态方法调用和使用通配符定义action
由请求方式来指定action执行方法。
1、动态方法调用
如果action存在多个方法时,我们可以使用!+方法名调用指定方法。如下:
public class HellWorldAction{
    private String message;
    public String execute()throws Exception{
        this.message="我的第一个struts应用";
        return "success";
    }
    public String other()throws Exception{
        this.message="我的第二个struts应用";
        return "success";
    }
}
假设访问上面action的URL路径为:/struts/test/helloworld.action
要访问action的other()方法,我们可以这样调用:
/struts/test/hellworld!other.action
如果不想使用动态方法调用,我们可以通过常量struts.enable.DynamicMethodInvocation关闭动态方法调用。
<constant name="struts.enable.DynamicMehodInvocation" value="false" />
******************************************************************************************************************

14:03 2012/3/24
关于struts2.1.6接收中文请求参数乱码问题(以post方式提交),原因是struts2.1.6在获取并使用了请求参数后才调用HttpServletRequest的setCharacterEncoding()方法进行编码设置,导致应用使用的就是乱码参数。这个bug在struts2.1.8中已经被解决,如果你使用的是struts2.1.6,要解决这个问题,你可以这样做:新建一个Filter,把这个Filter放置在struts2的Filter之前,然后再doFilter()方法里面添加以下代码:
public void doFilter(){
    HttpServletRequest req = (HttpServletRequest)request;
    req.setCharacterEncoding("UTF-8");//应该根据你使用的编码替换UTF-8
    filterchain.doFilter(request,response);
}
******************************************************************************************************************


14:10 2012/3/25
自定义类型转换
struts2有两种类型转换器
1、局部转换器
1.1、自定义类型转换器
public class DateTypeConverter extends DefaultTypeConverter {

    @Override
    public Object convertValue(Map<String, Object> context, Object target,
            Member member, String propertyName, Object value, Class toType) {
        SimpleDateFormat sf = new SimpleDateFormat("yyyyMMdd");
        try {
            if (toType == Date.class) {
                String[] param = (String[]) value;
                return sf.parse(param[0]);
            }else if(toType == String.class){
                Date date = (Date) value;
                return sf.format(date);
            }
        } catch (ParseException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return super.convertValue(context, target, member, propertyName, value,
                toType);
    }

1.2、将上面的类型转换器注册为局部类型转换器:
在action类所在的包下放置ActionClassName-conversion.properties文件,ActionClassName是Action的类名,后面的-conversion.properties是固定写法,对于本例而言,文件的名称应为HelloWorldAcion-conversion.properties。
在properties文件中的内容为:
属性名=类型转化器的全类名
对于本例而言,HelloWorldAcion-conversion.properties文件中的内容为:
createtime=cn.flame.conversion.DateConverter
2、全局转换器
全局转换器的代码内容与局部转换器内容一致,只是注册方式不同。
注册方式为:
在web-inf/classes下放置xwork-conversion.properties文件。在peoperties文件的内容为:
待转换的类型=类型转换器的全类名
对于本例而言,xwork-conversion.properties文件中的内容为:
java.util.Date=cn.flame.conversion.DateConverter
******************************************************************************************************************

16:24 2012/3/25
访问或添加request/session/application属性
action:
----------------------------------------------------------
public String execute(){
        ActionContext atc = ActionContext.getContext();
        atc.getSession().put("session", "session范围");
        atc.getApplication().put("application", "application范围");
        atc.put("request", "request范围");
        return "message";
    }
JSP:
----------------------------------------------------------
<body>
    ${applicationScope.application}</br>
    ${sessionScope.session}</br>
    ${requestScope.request}</br>
</body>
----------------------------------------------------------
******************************************************************************************************************

21:25 2012/3/26
获取HttpServletRequest/HttpSession/ServletContext/HttpServletResponse对象
方法一:通过ServletActionContext类直接获取:
public String rsa()throws Exception{
    HttpServletRequest request = ServletActionContext.getRequest();
    ServletContext servletContext = ServletActionContext.getServletContext();
    request.getSession();
    HttpServletResponse response = ServletActionContext.getResponse();
    return "scope";
}
方法二:实现接口,由struts框架运行时注入:
******************************************************************************************************************

0:27 2012/3/27
struts 中实行文件上传
第一步:在WEB-INF/lib下加入common-fileupload-1.2.1.jar、commons-io-1.3.2.jar。这两个文件可以从http://commons.apache.org/下载。
第二步:把form表的enctype设置为:"multipart/form-data",如下:
<form entctype="multipart/form-data" action="${pageContext.request.contextPath}/xxx.action" method="past">
    <imput type="file" name="uploadImage">
</form>
第三步:在action类中添加以下属性,属性名称应对应表单中文件字段的名称:
public class HelloWorldAction{
    private File uploadImage;//得到上传的文件
    private String uploadImageContextType;//得到文件的类型(为字段名+ContextType固定写法)
    private String uploadImageFileName;//得到文件的名称(为字段名+FileName固定写法)
    //此处省略了字段属性
    public String upload()throws Exception{
        String realpath = ServletActinoContext.getServletContext().getRealPath("/images");
        File file = new File(realpath);
        if(!file.exists()) file.makdirs();
        FileUtils.copyFile(uploadImage,new File(file,uploadImageFileName));
        return "success";
    }
}
******************************************************************************************************************

22:21 2012/3/27
多文件上传
第一步:在WEB-INF/lib下加入common-fileupload-1.2.1.jar、commons-io-1.3.2.jar。这两个文件可以从http://commons.apache.org/下载。
第二步:把form表的enctype设置为:"multipart/form-data",如下:
<form entctype="multipart/form-data" action="${pageContext.request.contextPath}/xxx.action" method="past">
    <imput type="file" name="uploadImage">
</form>
第三步:在action类中添加以下属性,属性名称应对应表单中文件字段的名称:
public class HelloWorldAction{
    private File[] uploadImage;//得到上传的文件
    private String[] uploadImageContextType;//得到文件的类型(为字段名+ContextType固定写法)
    private String[] uploadImageFileName;//得到文件的名称(为字段名+FileName固定写法)
    //此处省略了字段属性
    public String execute() {
        
            String realPath = ServletActionContext.getServletContext()
                    .getRealPath("/images");
            System.out.println(realPath);
            System.out.println(uploadFiles);
            System.out.println(uploadFilesFileName);
            ArrayList<String> arrayList = new ArrayList<String>();
            if (uploadFiles != null) {
                File savedir = new File(realPath);
                if (!savedir.exists())
                    savedir.mkdirs();
                for (int i = 0; i < uploadFiles.length; i++) {
                    File savefile = new File(savedir, uploadFilesFileName[i]);
                    FileUtils.copyFile(savefile, uploadFiles[i]);
                    arrayList.add(uploadFilesFileName[i] + "上传成功!</br>");
                }
                ActionContext.getContext().put("messageList", arrayList);
            }
        return "success";
    }
}
******************************************************************************************************************

23:45 2012/3/27
自定义拦截器
1、要自定义拦截器需要实现com.opensymphony.xwork2.interceptor.Interceptor接口:
invocation.invoke();执行被拦截方法。
2、注册拦截器
<package name="flame" namespace="/test" extends="struts-default" >
    <interceptors>
        <interceptor name="permission" class="cn.flame.aop.PermissionInterceptor" />
        <interceptor-stack name="permissionStack">
            <interceptor-ref name="defaultStack" />
            <interceptor-ref name="permission" />
        </interceptor-stack>
    </interceptors>
    <action name="hellworld" class="cn.flame.action.HelloWorldAction" method="{1}" >
        <result name="success">/WEB-INF/page/hello.jsp</result>
        <interceptor-ref name="permissionStack" />
    </action>
</package>
因为struts2中如文件上传、数据验证、封装请求参数到action等功能都是由系统默认的defaultStack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultStack。这样应用才可以使用struts2框架提供的众多功能。
如果希望包下的所有action都使用自定义的拦截器,可以通过<default-interceptor-ref name="permissionStack" />把拦截器定义为默认拦截器。
注意:每一个包只能定义一个默认拦截器,另外,一旦我们为该包中的某个action显示制定了某个拦截器,则默认拦截器不会起作用。
******************************************************************************************************************

0:33 2012/3/28
输入校验
在struts2中,我们可以实现对action的所有方法进行校验或者对action的指定方法进行校验。

对于输入校验struts2提供了两种实现方法:
1、采用手工编写代码实现;
2、基于XML配置方式实现;
******************************************************************************************************************

0:50 2012/3/28
手工重写代码实现对action中所有方法输入校验:
通过重写validate()方法实现,validate()方法会校验action中所有与execute方法签名相同的方法。当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息(为了使用addFildError()方法,action可以继承ActionSupport),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result。在input视图中可以通过<s:fielderror />显示失败信息。
validate()使用例子:
public void validate(){
    if(this.mobile==null ||"".equals(this.mobile.trim())){
        this.addFieldError("username","手机号码不能为空");
    }else{
        if(!Patternmcompile("^1[358]\\d{9}").matcher(this.mobile.trim()).matches()){
            this.addFieldError("mobile","手机号码格式不正确");
        }
    }
}
验证失败后,请求转发至input视图:
<result name="input" >/WEB-INF/page/addUser.jsp</result>
在addUser.jsp页面中使用<s:fielderror/>显示失败信息。
注:需要引入struts2标签:<%@ taglib uri="/struts-tags" prefix="s" %>
******************************************************************************************************************

9:37 2012/3/28
手工编写代码实现action指定方法输入校验
通过validateXxx()方法实现,validatexxx()指挥校验action中方法名为Xxx的方法。
其中Xxx的第一个字母要大些。
当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加失败信息(为了使用addFieldError()方法,action可以继承ActionSupport),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result。在input视图中可以通过<s:fielderror/>显示失败信息。
validateXxx()方法使用例子:
public String add() throws Exception{return "success";}
public void validateAdd(){

    if(username==null &&"".equals(username.trim())) this.addFieldError("username","用户名不能为空");
}
验证失败后,请求转发至input视图:
<result name="input">/WEB-INF/lib/page/addUser.jsp</result>
在addUser.jsp页面中使用<s:fielderror/>显示失效信息。
******************************************************************************************************************

10:28 2012/3/28
输入校验的流程
1、类型转换器对请求参数执行类型转换,并把转换后的值赋给action中的属性。
2、如果在执行类型转换过程中出现异常,系统会将异常信息保存到ActionContext,conversionError拦截器将异常信息封装到fieldErrors里。不管类型转换是否出现异常,都会进入第三步。
3.系统通过反射技术先调用action中的validateXxx()方法,Xxx为方法名。
4、再调用action中validate()方法。
5、经过上面4步,如果系统中的fieldErrors存在错误信息(即存放错误信息的集合的size大于0),系统自动将请求转发至名为input的视图。如果系统中的fieldErrors没用任何错误信息,系统将执行action中的处理方法。
******************************************************************************************************************

11:22 2012/3/28
基于XML配置方式实现action所有输入校验
使用基于xml配置方式实现输入校验是,action也需要继承ActionSupport,并且提供校验文件,校验文件盒action类放在同一个包下,文件的取名格式为:ActionClassName-validation.xml,其中ActionClassName为action的简单类名,-validation为固定写法。如果action类为cn.flame.UserAction,那么该文件的取名为:UserAction-validation.xml。下面是校验文件的模板:
-----------------------------------------
<!DOCTYPE validators PUBLIC
        
"-//Apache Struts//XWork Validator 1.0.3//EN"
        
"http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd">


<validators>
    
    <field name="username">
        
        <field-validator type="requiredstring">
            
            <message key="requiredstring"/>
        
        </field-validator>
    
    </field>
    
    <field name="password">
        
        <field-validator type="requiredstring">
            
            <message key="requiredstring"/>
        
        </field-validator>
    
    </field>

</validators>
------------------------------------------
<field>制定action中要校验的属性,<field-validator>制定校验器,上卖弄指定的校验器requiredstring是由系统提供的,系统提供了能满足大部分验证需求的校验器,这些校验器的定义可以再xwork-2.x.jar中的
com.opensymphony.xwork2.validator.validators下得default.xml中找到。
<message>为校验失败后的提示信息,如果需要国际化,可以为message指定key属性,key的值为资源文件中的key。
在这个校验文件中,对action中字符串类型的username属性进行验证,首先需要调用trim()方法去掉空格,然后判断用户名是否为空。
******************************************************************************************************************

13:59 2012/3/28
校验器的使用例子
---------------------------------------------------------------------
required必填校验器
---------------------------------------------------------------------
<field-validator type="required">
            
    <message>性别不能为空</message>
        
</field-validator>
---------------------------------------------------------------------
requiredstring必填字符串校验器
---------------------------------------------------------------------
<field-validator type="requiredstring">
            
    <param name="trim">true</param>
    <message>用户名不能为空</message>
        
</field-validator>
---------------------------------------------------------------------
stringlength字符串长度校验器
---------------------------------------------------------------------
<field-validator type="stringlength">
            
    <param name="maxLength">10</param>
    <param name="minLength">2</param>
    <param name="trim">true</param>
    <message><![CDATA[产品名称应在2-10个字符之间]]></message>
        
</field-validator>
---------------------------------------------------------------------
email邮件地址校验器
---------------------------------------------------------------------
<field-validator type="email">
            
    <message>电子邮箱地址无效</message>
        
</field-validator>
---------------------------------------------------------------------
regex正则表达式校验器
---------------------------------------------------------------------
<field-validator type="regex">
            
    <param name="expression"><![CDATA[^1[358]\D{9}$]]></param>
    <message>手机号格式不正确</message>
        
</field-validator>
******************************************************************************************************************

14:16 2012/3/28
编写校验文件时,不能出现帮助信息解决方法:
在编写ActionClassName-validation.xml校验文件时,如果出现不了帮助信息,可以按下面方式解决:
window ->preferences->myeclipse->files and editors -> xml -> xmlcatalog
点“add”,在出现的窗口中的location中选“File system",然后在xwork-2.1.2解压目录的src\java目录中选择xwork-validator-1.0.3.dtd,回到设置窗口的时候不要急着关闭窗口,应把窗口中的 Key Type改为URL。key改为
http://www.opensymphony.com/xwork/xwork-validator-1.0.3.dtd
******************************************************************************************************************

14:36 2012/3/28
基于XML配置方式对指定action方法实现输入校验
当校验问及爱你取名为ActionClassName-validation.xml时,会对action中的所有处理方法实施输入验证。如果你只需要对action中的某个action方法实施校验,那么校验文件的取名应为:ActionClassName-ActionName-validation.xml,其中ActionName为struts.xml中的action的名称。
例如:在实际应用中,常有一下配置:
-----------------------------------------------------------------------------------
<action name="user_*" class="cn.flame.action.UserAction" method="{1}">
    <result name="success">/WEB-INF/page/message.jsp</result>
    <result name="input">/WEB-INF/page/addUser.jsp</result>
</action>
-----------------------------------------------------------------------------------
UserAction中有一下两种处理方法:
public String add() throws Exception{
    .....
}
public String update() throws Exception{
    .....
}
-----------------------------------------------------------------------------------
要对add()方法实施验证,校验文件的取名为:UserAction-user_add-validation.xml
要对update()方法实施验证,校验文件的取名为:UserAction-user_update-validation.xml
******************************************************************************************************************

14:49 2012/3/28
基于XML校验的一些特点:
两种规则的校验文件时,系统按下面顺序寻找校验文件:
1、ActionClassName-validation.xml
2、ActionClassName-ActionName-validation.xml
系统寻找到第一个校验文件时还会继续搜索后面的校验文件,当搜索到所有校验文件时,会把校验文件里的所有校验规则汇总,然后全部应用于action方法的校验。如果两个校验文件中指定的校验规则冲突,则只使用后面文件中的校验规则。
当action继承了另一个action,父类action的校验文件会先被搜索到。
假设UserAction继承BaseAction:
<action name="user" class="cn.flame.acion.UserAction" method="{1}">
</action>
访问上面action,系统先搜索父类的校验文件:BaseAction-validation.xml,BaseAction-user-validation.xml,接着搜索子类的校验问及爱你:UserAction-validation.xml,UserAction-user-validation.xml。应用于上面action的校验规则为这四个文件的总和。
******************************************************************************************************************

15:11 2012/3/28
配置国际化全局资源文件、输出国际化信息
准备资源文件,资源文件的命名格式如下:
baseName_language_country.properties
baseName_language.properties
baseName.properties
其中baseName是资源文件的基本名,我们可以自定义,但language和country必须是java支持的语言和国家。如:
中国大陆:baseName_zh_CN.properties
美国:baseName_en_US.properties

现在为应用添加两个资源文件:
第一个存放中文:itcast_zh_CN.properties
内容为:welcome=欢迎来到传智播客
第二个存放英语(美国):itcast_en_US.properites
内容为:welcome=welcome to itcast

对于中文的属性文件,我们编写好后,应该使用jdk提供的native2ascii命令把文件转换为unicode编码的文件。命令的使用方式如下:
native2ascii 源文件.properties 目标文件.properties

当准备好资源文件之后,我们可以在struts.xml中通过struts.custom.i18n.resources常量把资源文件定义为全局资源文件,如下:
<constant name="struts.custom.i18n.resources" value="flame" />
flame为资源文件的基本名。

后面我们就可以在页面或action中访问国际化信息:
在jsp页面中使用<s:text name="" />资源输出国际化信息:
<s:text name="user" />,name为资源文件中的key
在Action类中,可以继承ActionSupport,使用getText()方法得到国际化信息,该方法的第一个参数用于指定资源文件中的key。
在表单标签中,可以通过key属性指定资源文件中的key,如:
<s:textfield name="realname" key="user" />
******************************************************************************************************************

16:57 2012/3/28
国际化——输出带占位符的国家化信息
资源文件中的内容如下:
welcome={0},欢迎来到传智播客{1}

在jsp页面中输出带占位符的国家化信息
<s:text name="welcome">
    <s:param><s:property value="realname" /></s:param>
    <s:param>学习</s:param>
</s:text>
在action类中获取带占位符的国际化信息,可以使用getText(String key,String[] args)或getText(String aTextName,List args)方法。
******************************************************************************************************************

17:09 2012/3/28
配置包范围国际化资源文件
在一个大型应用中,整个应用有大量的内容需要实现国际化,如果我们把国家化的内容都放置在一个全局资源属性文件中,显然会导致资源文件变得过于庞大、臃肿,不便于维护,这个时候我们可以针对不同模块,使用包方位来组织国家化文件。

方法如下:
在java的包下放置package_language_country.properties资源文件,package为固定写法,处于该包及子包下得action都可以访问该资源。当查找指定key的消息是,系统会先从package资源文件查找,当找不到对应的key时,才会从常量struts.custom.i18n.resources指定的资源文件中寻找。
******************************************************************************************************************

17:24 2012/3/28
配置action范围资源文件
我们也可以在某个action单独指定资源文件,方法如下:
在action类所在的路径,放置ActionClassName_lanuage_country.properties资源文件,ActionClassName为action类的简单名称。
******************************************************************************************************************

17:59 2012/3/28
国际化——jsp中直接访问某个资源文件
struts2为我们提供了<s:i18n/>标签,使用<s:i18n/>标签我们可以在类路径下直接从某个资源文件中获取国际化数据,而无需任何配置:
<s:i18n name="flame">
    <s:text name="welcome" />
</s:i18n>
flame为类路径下资源文件的基本名。

如果要访问资源文件在类路径的某个包下,可以这样访问:
<s:i18n name="cn/flame/action/package">
    <s:text name="welcome" />
        <s:param >小张</s:param >
</s:i18n>
上面访问cn.flame.action包下基本名为package的资源文件。
******************************************************************************************************************

20:42 2012/3/28
OGNL表达式语言
OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目。Struts2框架使用OGNL作为默认的表达式语言。

相对EL表达式,它提供了平时我们需要的一些功能,如:
1、支持对象方法调用,如xxx.sayHello();
2、支持类静态方法调用和值访问,表达式的格式为@[类全名(包括包路径)]@[方法名|值名]
例如:
@java.lang.String@format('foo%s','bar')

@cn.flame.Constant@APP_NAME;
3、操作集合对象

OGNL有一个上下文(context)概念,说白了上下文就是一个MAP结构,它实现了java.utils.map接口,在struts2中上下文(context)的实现为ActionContext,下面是上下文(context)的结构示意图:

        |-----ValueStack(值栈,它是跟对象)
        |
        |-----parameters
        |
        |-----request
OGNL Context----|
        |-----session
        |
        |-----application
        |
        |-----attr

当struts2接受一个请求时,会迅速创建ActionContext,ValueStack,action。然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问。

访问上下文(context)中的对象需要使用#符号标注命名空间,如#application、#session

另外OGNL会设定一个根对象(root对象),在struts2中跟对象就是ValueStack(值栈)。如果要访问跟对象(即ValueStack)中对象的属性,则可以省略#命名空间,直接访问该对象的属性即可。

在struts2中,跟对象ValueStack的实现类为OgnlValueStack,该对象不是我们想象中的只存放单个值,而是存放一组对象。在OgnlValueStack类里有一个List类型的root变量,就是使用它存放一组对象。

        |-----request
        |
        |-----application
        |
        |-----OgnlValueStack root变量[action,OgnlUtil,。。。]
     Context----|
        |-----session
        |
        |-----attr
        |
        |-----parameters

在root变量中处于第一位的对象叫栈对象。通常我们再OGNL表达式里直接写上属性的名称即可访问root变量里对象的属性,搜索顺序是从栈顶对象开始寻找,如果栈顶对象不存在该属性,就会从第二个对象寻找,如果没有找到就从第三个对象寻找,依次往下访问,知道找到为止。

注:struts2中,OGNL表达式需要配合struts标签才可以使用。如:<s:property value="name" />

由于ValueStack是struts2中OGNL的跟对象,如果用户需要访问值栈中的对象,在jsp页面可以直接通过下面的EL表达式访问ValueStack中对象的属性:
${foo}//获得值栈中某个对象的foo属性

如果访问其他context中的对象,由于他们不是跟对象,所以在访问时,需要添加#前缀。
1、application对象:用于访问servletContext,例如#application.userName或者#application['userName'],相当于调用ServletContext的getAttribute("userName")。

2、session对象,用来访问HttpSession。例如#session.userName或者#session['userName'],相当于调用session。getAttribute("userName").

3、request对象:用来访问HttpServletRequest属性(attribute)的map,例如#request.userName或者#request['userName'],相当于调用request.getAttribute("userName")。

4、parameters对象:用于访问HTTP的请求参数,例如#parameters.userName或者#parameter['userName'],相当于调用request.getParameter('userName')。

5、attr对象:用于按page ->request ->session ->application顺序访问其属性。

******************************************************************************************************************

22:06 2012/3/28
为何使用EL表达式能够访问valueStack中对象的属性
原因是struts2对HttpServletRequest作了进一步的封装。
******************************************************************************************************************

采用OGNL表达式创建List/Map集合对象
如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中通集合相关的表达式,使用如下代码直接生成一个List对象:
----------------------------------------------------------------------------
    <s:set name="list" value="{'zhangzhanlong','qiuweike','wanli'}" />
    <s:iterator value="#list">
        <s:property/>
    </s:iterator>
----------------------------------------------------------------------------
Set标签用于将某个值放入指定范围。
scope:指定变量被放置的范围ie,该属性可以接受application、session、request、page或action。如果没有设置该属性,则默认放置在OGNL Context中。
value:赋给变量的值,如果没有放置该属性,则将ValueStack栈顶的值赋给变量。

生成一个Map对象:
----------------------------------------------------------------------------
    <s:set name="maps" value="#{'key1':33,'key2':44,'key3':55}" />
    <s:iterator value="#maps">
        <s:property value="key"/>=<s:property value="value "/>
    </s:iterator>
----------------------------------------------------------------------------
******************************************************************************************************************


23:43 2012/3/28
采用OGNL表达式判断对象是否存在于集合中
对于集合类型,OGNL表达式可以使用in和not in两个元素符号。
in表达式用来判断某个元素是否在指定的集合对象中;
not in判断某个元素是否不再指定的集合对象中,如下所示;
    <s:if test="'foo' in {'foo','fss'}">
        foo在
    </s:if>
    <s:else>
        foo不在
    </s:else>
    <s:if test="'foo' not in {'foos','fss'}">
        foo不在
    </s:if>
    <s:else>
        foo在
    </s:else>

******************************************************************************************************************

23:47 2012/3/28
OGNL表达式的投影功能
除了in和not in之外,OGNL还允许使用某个规则获得集合对象的子集,常用的有以下3个相关操作符。
?:获得所有符合逻辑的元素;
^:获得符合逻辑的第一个元素;
$:获得符合逻辑的最后一个元素;
例如代码:
    <s:iterator value="books.{?#this.price >35}">
        <s:property value="title"/>-<s:property value="price"/>        
    </s:iterator>
在上面中,直接在集合后紧跟.{}运算符表明用于取出该集合的子集,{}内的表达式用于获取符合条件的元素,this值得是为了从大集合books筛选数据到小集合,需要对大集合books进行迭代,this代表当前迭代的元素。本例的表达式用于获取集合中价格大于35的子集合。
******************************************************************************************************************

0:03 2012/3/29
struts2常用标签

一、property标签
property标签用于输出指定值:
<s:property value="#session.password"/>
default:可选属性,如果需要输出的属性值为null,则显示该属性指定的值。
escape:可选属性,指定是否格式化HTML代码。
value:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值。
id:可选属性,指定该元素的标识

二、iterator标签用于对集合进行迭代,这里的集合包含List、Set和数组。
    <s:set name="maps" value="#{'key1':33,'key2':44,'key3':55}" />
    <s:iterator value="#maps">
        <s:property value="key"/>=<s:property value="value "/>
    </s:iterator>
value:可选属性,指定被迭代的集合,如果没有设置该属性,则使用ValueStack栈顶的集合。
id:可选属性,指定集合里元素的id(已经被标注过时)
status:可选属性,该属性指定迭代式的IteratorStatus实例。该实例包含如下几个方法:
    int getCount(),返回当前迭代了几个元素
    int getIndex(),返回当前迭代元素的索引
    boolean isEven(),返回当前被迭代元素的索引是否是偶数
    boolean isOdd(),返回当前被迭代的索引是否是奇数
    boolean isFirst(),返回当前被迭代元素是否是第一个元素
    boolean isLast(),返回当前被迭代元素是否是最后一个元素

三、if/elseif/else标签
    <s:if test="'foo' in {'foo','fss'}">
        foo在
    </s:if>
    <s:else>
        foo不在
    </s:else>
    <s:if test="'foo' not in {'foos','fss'}">
        foo不在
    </s:if>
    <s:else>
        foo在
    </s:else>
四、url标签
<s:url action="helloworld_add" namespace="/test">
    <s:param name="personid" value="23" />
</s:url>
生成类似如下路径:
/struts/test/helloworld_add.action?personid=23

当标签的属性值作为字符串类型处理时,"%"符号的用途是计算OGNL表达式的值。
<s:set name="myurl" value="http://www.foshanshop.net" />
<s:url value="#myurl" /><br/>
<s:url value="%{#myurl}" />
输出结果:
#myurl
http://www.foshanshop.net

五、表单标签_checkboxlist复选框
如果集合为list:
<s:checkboxlist name="list" list="{'java','net','c#'}" value="{'java','c#'}"></s:checkboxlist>
页面执行完会自动生成如下代码:
<input type="checkbox" name="list" value="java"
       id="list-1"       checked="checked"        />
<label for="list-1" class="checkboxLabel">java</label>
<input type="checkbox" name="list" value="net"
       id="list-2"        />
<label for="list-2" class="checkboxLabel">net</label>
<input type="checkbox" name="list" value="c#"
       id="list-3"       checked="checked"        />
<label for="list-3" class="checkboxLabel">c#</label>
--------------------------------------------------------------
如果集合为MAP:
<s:checkboxlist name="map" list="#{1:'java',2:'net',3:'c#'}" value="{1,2,3}" listKey="key" listValue="value"></s:checkboxlist>
执行完生成代码:
<input type="checkbox" name="map" value="1"
       id="map-1"       checked="checked"        />
<label for="map-1" class="checkboxLabel">java</label>
<input type="checkbox" name="map" value="2"
       id="map-2"       checked="checked"        />
<label for="map-2" class="checkboxLabel">net</label>
<input type="checkbox" name="map" value="3"
       id="map-3"       checked="checked"        />
<label for="map-3" class="checkboxLabel">c#</label>

六、单选框的使用,与复选框雷同。
七、下拉框的使用,与复选框雷同。
******************************************************************************************************************

0:40 2012/3/29
struts2中防止表单重复提交
<s:token />标签防止重复提交,用法如下:
第一步:在表单中加入<s:token />
<s:form action="helloworld_other" method="post" namespace="/test" >
    <s:textfield name="person.name" /> <s:token /> <s:submit/>
</s:form>
第二步:
<action name="flame" class="cn.flame.action.FlameAction" >
    <interceptor-ref name="defaultStack" />
    <interceptor-ref name="token" />
    <result name="invalid.token">/index.jsp</result>
    <result name="success">/WEB-INF/page/message.jsp</result>
</action>
以上配置加入了”token“拦截器和”invalid.token“结果,因为”token“拦截器在会话的token与请求的token不一致时,将会直接返回”invalid.token“结果。

在debug状态,控制台出现异常信息是因为action中并没有struts.token和struts.token.name属性,我们不用关心这个错误。
******************************************************************************************************************
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
前台: (1)注册登录模块:按照学校的相关规定进行注册和登录。 (2)招聘信息查看:高校毕业生们可以网站首页上查看所有的招聘信息,除此之外还可以输入公司名称或岗位名称进行搜索。 (3)用人单位模块:此模块为宣传用人单位的主要功能模块,具体包括用人单位简介、岗位需求及职责及公司介绍等功能。 (4)就业指导:学生朋友们在就业前可以通过此模块获取指导。 (5)新闻信息:为了让用户们可以了解到最新的新闻动态,本系统可以通过新闻信息查看功能阅读近期的新闻动态。 (6)在线论坛:毕业季的同学们可以通过此模块相互交流。 后台: (1)系统用户管理模块:可以查看系统内的管理员信息并进行维护。 (2)学生管理模块:通过此功能可以添加学生用户,还可以对学生信息进行修改和删除。 (3)用人单位管理模块:管理员用户通过此模块可以管理用人单位的信息,还可以对用人单位信息进行查看和维护。 (4)招聘管理模块:管理员通过此功能发布和维护系统内的照片信息。 (5)就业指导管理模块:通过此模块可以编辑和发布就业指导信息,从而更好的帮助就业季的同学们。 (6)论坛管理:通过论坛管理可以查看论坛中的主题帖及里面的回复信息,除此之外还可以对论坛中的信息进行维护和管理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值