Struts2学习第二天——获取参数与数据校验

文档版本 开发工具 测试平台 工程名字 日期 作者 备注
V1.0 2016.06.14 lutianfei none

struts2中获取请求参数

  • 在struts2中action是什么?(struts2是一个mvc框架)

    • View : jsp
    • Model : action
    • Control : action & StrutsPrepareAndExecuteFilter

1.属性驱动

  • 1.直接将action做一个model(类似bean结构),就可以得到请求参数.

    • 问题1:action封装请求参数,会不会存在线程安全问题?

      • 不会:因为每一次请求,都是一个新的action。
      • 缺点:需要单独定义javaBean,将action中属性copy到javaBean中。
      • 优点:简单。
      • 这种方式,底层是通过反射来实现的。
  • 配合第一天作业案例,修改Login1Action.java

    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;
    }

    @Override
    public String execute() throws Exception {
        HttpServletRequest request = ServletActionContext.getRequest();
        // 2.判断用户名与密码是否正确
        if ("tom".equals(username) && "123".equals(password)) {

            request.getSession().setAttribute("username", username);

            return SUCCESS;
        } else {
            request.setAttribute("login.message", "用户名或密码错误");
            return "failer";
        }
    }
}
  • 2.在action中声明一个model。

    • 在页面上使用ognl来描述
    • 注意修改:<input type="text" name="user.username">
      • 优点:简单,解决了第一种封装的问题
      • 缺点:在页面上使用了ognl表达式,页面不通用了。
    • 问题:这种方式,数据是怎样封装的?
      • 通过struts2中的interceptor进行了数据封装.<interceptor name="params" class="com.opensymphony.xwork2.interceptor.ParametersInterceptor"/>
  • Login2Action.java
//获取请求参数  属性驱动  第二种,直接在action声明一个model
public class Login2Action extends ActionSupport {

    private User user;

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    @Override
    public String execute() throws Exception {
        HttpServletRequest request = ServletActionContext.getRequest();
        // 2.判断用户名与密码是否正确
        if ("tom".equals(user.getUsername()) && "123".equals(user.getPassword())) {

            request.getSession().setAttribute("username", user.getUsername());

            return SUCCESS;
        } else {
            request.setAttribute("login.message", "用户名或密码错误");
            return "failer";
        }
    }
}
  • login2.jsp
  <body>
      ${requestScope["login.message"] }<br>
    <form action="${pageContext.request.contextPath}/login2" method="post">
        username:<input type="text" name="user.username"><br>
        password:<input type="password" name="user.password"><br>
        <input type="submit" value="登录">
    </form>
  </body>

2.模型驱动(在开发中应用比较多)

  • 开发步骤:

    • 1.让action类实现 ModelDriven
    • 2.重写getModel方法
    • 3.在action中实例化一个model对象,让getModel方法返回这个对象。
public class Login3Action extends ActionSupport implements ModelDriven<User> { //第一步:implements ModelDriven<User>

private User user = new User(); //第三步

public User getModel() { //第二步
return user;
}

}

//jsp
  <body>
      ${requestScope["login.message"] }<br>
    <form action="${pageContext.request.contextPath}/login3" method="post">
        username:<input type="text" name="username"><br>
        password:<input type="password" name="password"><br>
        <input type="submit" value="登录">
    </form>
  </body>
  • 优点:解决了属性驱动存在的问题
  • 缺点:一次只能封装一个model对象.
  • struts2 有很多围绕模型驱动的特性
    • <interceptor name="modelDriven" class="com.opensymphony.xwork2.interceptor.ModelDrivenInterceptor"/> 为模型驱动提供了更多特性

扩展

  • 1.将数据封装到List集合

页面:

username1:<input type="text" name="users[0].username"><br>
password1:<input type="password" name="users[0].password"><br>

username2:<input type="text" name="users[1].username"><br>
password2:<input type="password" name="users[1].password"><br>

action类:

public class ListAction extends ActionSupport {

    private List<User> users;

    public List<User> getUsers() {
        return users;
    }

    public void setUsers(List<User> users) {
        this.users = users;
    }

    @Override
    public String execute() throws Exception {

        System.out.println(users);

        return null;
    }
}

//struts.xml配置
<action name="list" class="cn.itcast.action.ListAction">
</action>
  • 2.将数据封装到Map集合

页面:

username1:<input type="text" name="map[‘aaa‘].username"><br>
password1:<input type="password" name="map[‘aaa‘].password"><br>

username2:<input type="text" name="map[‘bbb‘].username"><br>
password2:<input type="password" name="map[‘bbb‘].password"><br>
  • action类:
public class MapAction extends ActionSupport {

    private Map<String, User> map;

    public Map<String, User> getMap() {
        return map;
    }

    public void setMap(Map<String, User> map) {
        this.map = map;
    }

    @Override
    public String execute() throws Exception {

        System.out.println(map);

        return null;
    }
}

struts2中提供的类型转换

  • 在web中我们使用beanutils直接将表单数据封装到javaBean中。—类型转换
  • struts2中action得到请求参数,也可以直接封装到javaBean.
  • struts2 内部提供大量类型转换器,用来完成数据类型转换问题
    • boolean 和 Boolean
    • char 和 Character
    • int 和 Integer
    • long 和 Long
    • float 和 Float
    • double 和 Double
    • Date 可以接收 yyyy-MM-dd格式字符串
    • 数组 可以将多个同名参数,转换到数组中
    • 集合 支持将数据保存到 List 或者 Map 集合
  • 但并不是所有的类型都支持
    • 例如:日期类型,我们传递 yyyy-MM-dd yyyy年MM月dd日格式都可以,但是如果是yyyy/MM/dd就会出现问题.
  • 关于struts2中的类型转换器根接口是:com.opensymphony.xwork2.conversion.TypeConverter

自定义类型转换器

  • 操作步骤:

    • 1.创建一个类实现TypeConverter接口
    • 2.重写接口中方法,实现类型转换操作
    • 3.注册类型转换器
1.创建一个自定义类型转换器
  • 1.实现TypeConverter需要重写如下方法:

    • public Object convertValue(Map<String, Object> context, Object target, Member member, String propertyName, Object value, Class toType);如果实现接口,这个方法参数太多(6个),不推荐。
  • 2.继承 DefaultTypeConverter类
    • 优点:重写的方法参数没有那么多
    • public Object convertValue(Map<String, Object> context, Object value, Class toType) {return convertValue(value, toType);}
  • 3.推荐使用 继承DefaultTypeConverter类的一个子类StrutsTypeConverter.
    • 原因:在这个类中将从页面传递的数据怎样封装,以及action中的数据怎样在页面上显示做了分离。
    • convertFromString : 将页面数据转换至JavaBean
    • convertToString : 将JavaBean转换至页面输出
    public abstract Object convertFromString(Map context, String[] values, Class toClass);
    public abstract String convertToString(Map context, Object o);
2.注册一个自定义类型转换器
  • 1.局部–针对于action(action中包含了一个Bean结构)

    • 配置文件所在位置以及名称: 在Action类所在包 创建 Action类名-conversion.properties
    • 配置文件书写格式 : 属性名称=类型转换器的全类名
  • eg:

birthday= test.utils.MyTypeConverter
  • MyTypeConverter.java
public class MyTypeConverter extends StrutsTypeConverter{

    @Override
    public Object convertFromString(Map context, String[] values, Class toClass) {
        // System.out.println(context); //[email protected]
        // System.out.println(values[0]); //1990/11/10
        // System.out.println(toClass); //class java.util.Date

        String  value = values[0];

        SimpleDateFormat sdf  =new SimpleDateFormat("yyyy/MM/dd");

        Date date = null;

        try {
            date = sdf.parse(value);
        } catch (ParseException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

        return date;

    }
  • RegistAction.java
public class RegistAction extends ActionSupport implements ModelDriven<User>{

    private User user = new User();

    @Override
    public User getModel() {
        //
        return user;
    }        

    @Override
    public String execute() {
        System.out.println(user);
        return null;
    }
}
  • sturts.xml & regist.jsp
<struts>
    <package name="default" namespace="/" extends ="struts-default">
        <action name="regist" class = "test.action.RegistAction">
            <result name="input">/success.jsp</result>
        </action>
    </package>
</struts>

<body>
    <form action="${pageContext.request.contextPath}/regist" method="post">
        username:<input type="text" name="username"><br>
        password:<input type="password" name="password"><br>
        hobby:<input type="checkbox" name="hobby" value="eat">吃<input
            type="checkbox" name="hobby" value="drink">喝<input
            type="checkbox" name="hobby" value="play">玩<br> age:<input
            type="text" name="age"><br> birthday:<input type="text"
            name="birthday"><br> <input type="submit" value="注册">
    </form>
</body>
  • 2.局部–针对于model(action中实例化了一个Bean)

    • 配置文件所在位置以及名称: 在model类所在包 创建 model类名-conversion.properties
    • 配置文件书写格式 : 属性名称=类型转换器的全类名
  • eg:在上例的基础上做如下修改
    • 修改properties文件名及位置如下:

  • RegistAction.jsp
public class RegistAction extends ActionSupport {
    private String username;
    private String password;
    private int age;
    private Date birthday;
    private String[] hobby;

    public String[] getHobby() {
        return hobby;
    }

    public void setHobby(String[] hobby) {
        this.hobby = hobby;
    }

    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 int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    @Override
    public String execute() throws Exception {

        System.out.println(this.toString());

        return null;
    }

    @Override
    public String toString() {
        return "RegistAction [username=" + username + ", password=" + password
                + ", age=" + age + ", birthday=" + birthday + ", hobby="
                + Arrays.toString(hobby) + "]";
    }

}
  • 3.全局

    • 配置文件所在位置以及名称:在src下创建一个xwork-conversion.properties
    • 配置文件书写格式: 要转换的类型全名=类型转换器的全类名
  • eg:
    • 在第一个例子的基础上改成指定名字,并将其放在src目录下就好。
    • properties文件改为:java.util.Date=test.utils.MyTypeConverter
  • 注意:

    • 对于struts2中类型转换器(非自定义),如果表单数据提交时,将数据向model封装,出现了问题,会报错:

      No result defined for action cn.itcast.action.RegistAction and result input

    • 意思是,在RegistAction的配置中没有配置input结果视图;当加入result name=input标签时,会跳转到对应的页面(例如success.jsp)
    <action name="regist" class="cn.itcast.action.RegistAction">
    <result name="input">/success.jsp</result>
    </action>
  • 为什么会向input视图跳转?

    • 因为struts2中的拦截器(interceptor).
    • 在struts2中的<interceptor name="conversionError" class="org.apache.struts2.interceptor.StrutsConversionErrorInterceptor"/> 用于记录类型转换问题。
    • 在struts2中<interceptor name="workflow" class="com.opensymphony.xwork2.interceptor.DefaultWorkflowInterceptor"/>用于得到问题,向input视图跳转。

  • 关于错误信息展示:

    • 通过分析拦截器作用,得知当类型转换出错时,自动跳转input视图 ,在input视图页面中 <s:fieldError/> 显示错误信息
  • 如果想通过国际化改为中文显示需要在Action所在包中,创建 ActionName.properties,在局部资源文件中配置提示信息 : invalid.fieldvalue.属性名= 错误信息
  • eg:
//注意加 taglib

<%@ taglib prefix="s" uri="/struts-tags" %>
  <body>
    <s:fielderror/>
  </body>

//RegisAction.properties
invalid.fieldvalue.birthday=\u65E5\u671F\u683C\u5F0F\u8981\u6C42 yyyy/MM/dd
  • 如果是自定义类型转换器,出现类型转换问题,要跳转到input视图,在类型转换器中,必须抛出异常才可以。

    • eg : MyTypeConverter:第20行
public class MyTypeConverter extends StrutsTypeConverter {

    // 接收页面传递的数据封装到javaBean.
    @Override
    public Object convertFromString(Map context, String[] values, Class toClass) {

        // System.out.println(context); //[email protected]
        // System.out.println(values[0]); //[Ljava.lang.String;@100c4d
        // System.out.println(toClass); //class java.util.Date

        String value = values[0];
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd");
        Date date = null;

        try {
            date = sdf.parse(value);
        } catch (ParseException e) {

            // e.printStackTrace();
            throw new RuntimeException();
        }

        return date;
    }

关于struts2提供的数据校验

  • 在开发中,请求参数是需要校验的。

    • 客户端校验—->js
    • 服务器校验—->java代码。
  • struts2中提供的校验—–服务器端校验。
  • 分成两种:
    • 1.手动校验(编码校验)
    • 2.配置校验(annotation,xml) 我们讲的是xml。

1.手动校验(了解)

  • 要求:action类必须继承自ActionSupport。因为需要重写一个方法 validate
  • 通过测试发现在action中重写的validate方法执行了。并且是在请求处理方法(execute)之前执行的。

  • struts2提供的校验,也是通过拦截器实现的。
  • 问题:在validate方法中怎样存储校验错误信息?
    • 在validate方法中 this.addFieldError(Sting name,String value);
    @Override
    public void validate() {

        if (user.getUsername() == null
                || user.getUsername().trim().length() == 0) {
            // 说明用户名为空

            this.addFieldError("username.message", "用户名不能为空");

        }

        if (user.getPassword() == null
                || user.getPassword().trim().length() == 0) {
            this.addFieldError("password.message", "密码不能为空");

        }

        if (!(user.getAge() >= 10 && user.getAge() <= 40)) {
            this.addFieldError("age.message", "年龄必须在10-40之间");
        }

        // System.out.println("validate......");
    }
  • 问题:在页面上怎样获取错误信息?(在input视图上)

    • <s:fielderror> 展示所有错误信息
    • <s:fielderror fieldName="">展示特定名称的错误信息.
    <s:fielderror/>
    <form action="${pageContext.request.contextPath}/regist" method="post">
        username:<input type="text" name="username"><s:fielderror fieldName="username.message"/><br>
        password:<input type="password" name="password"><s:fielderror fieldName="password.message"/><br>
  • 问题:在同一个Action中有多个请求处理方法(login,regist)那么有些方法是需要校验的,有些是不需要的,怎样处理?

    • 解决方案:创建一个名称叫 validate+请求处理方法名
    • 例如:请求处理方法叫 regist() 校验的 方法名 validateRegist()
    • 如上情况下还有validate方法时,会先运行validateRegist()方法,后运行validate()方法。

2.配置校验(xml,常用)

  • struts2的校验框架。
  • 已经完成了校验操作(做了很多校验方法)。而我们在使用时,只需要将它们调用就可以(通过配置文件)
  • 要求:action类必须继承自ActionSupport类。
校验配置文件的配置方法
  • 位置:xml文件要与action类同一个包下
  • 名称: action类名-validation.xml
  • 约束: xwork-core-2.3.7.jarxwork-validator-1.0.3.dtd
<!DOCTYPE validators PUBLIC
"-//Apache Struts//XWork Validator 1.0.3//EN"
"http://struts.apache.org/dtds/xwork-validator-1.0.3.dtd">

  • 书写:

    • 1.根元素 <validators>
    • 2.子元素 <field name="属性名称"></field>
    • 3.<field>子元素
      • 指定校验器<field-validator type="校验器">
      • 问题:校验器有哪些?
      • 校验器位置:xwork-core-2.3.7.jar/com/opensymphony/xwork2/validator/validators/default.xml下
<validator name="required" class="com.opensymphony.xwork2.validator.validators.RequiredFieldValidator"/>
<validator name="requiredstring" class="com.opensymphony.xwork2.validator.validators.RequiredStringValidator"/>
<validator name="int" class="com.opensymphony.xwork2.validator.validators.IntRangeFieldValidator"/>
<validator name="long" class="com.opensymphony.xwork2.validator.validators.LongRangeFieldValidator"/>
<validator name="short" class="com.opensymphony.xwork2.validator.validators.ShortRangeFieldValidator"/>
<validator name="double" class="com.opensymphony.xwork2.validator.validators.DoubleRangeFieldValidator"/>
<validator name="date" class="com.opensymphony.xwork2.validator.validators.DateRangeFieldValidator"/>
<validator name="expression" class="com.opensymphony.xwork2.validator.validators.ExpressionValidator"/>
<validator name="fieldexpression" class="com.opensymphony.xwork2.validator.validators.FieldExpressionValidator"/>
<validator name="email" class="com.opensymphony.xwork2.validator.validators.EmailValidator"/>
<validator name="url" class="com.opensymphony.xwork2.validator.validators.URLValidator"/>
<validator name="visitor" class="com.opensymphony.xwork2.validator.validators.VisitorFieldValidator"/>
<validator name="conversion" class="com.opensymphony.xwork2.validator.validators.ConversionErrorFieldValidator"/>
<validator name="stringlength" class="com.opensymphony.xwork2.validator.validators.StringLengthFieldValidator"/>
<validator name="regex" class="com.opensymphony.xwork2.validator.validators.RegexFieldValidator"/>
<validator name="conditionalvisitor" class="com.opensymphony.xwork2.validator.validators.ConditionalVisitorFieldValidator"/>
  • 4.<field-validator>子元素

    • <message>错误信息</message>
  • 5.<field-validator>子元素
    • <param name="">值</param> : 用于指定校验器中的参数。
  • eg :
<validators>
    <!-- 对username属性进行校验 -->
    <field name="username">
        <!-- 指定username不能为空 -->
        <field-validator type="requiredstring">
            <!-- 错误信息 -->
            <message>用户名不能为空--------</message>
        </field-validator>
</validators>
配置校验中的常用校验器
  • required (必填校验器,要求被校验的属性值不能为null)
  • requiredstring (必填字符串校验器,要求被校验的属性值不能为null,并且长度大于0,默认情况下会对字符串去前后空格),如果不想去空格,只需如下设置。

  • stringlength (字符串长度校验器,要求被校验的属性值必须在指定的范围内,否则校验失败
    • minLength参数指定最小长度
    • maxLength参数指定最大长度
    • trim参数指定校验field之前是否去除字符串前后的空格

  • regex 正则表达式校验器,检查被校验的属性值是否匹配一个正则表达式,expression参数指定正则表达式,caseSensitive参数指定进行正则表达式匹配时,是否区分大小写,默认值为true)

  • int(整数校验器,要求field的整数值必须在指定范围内
    • min指定最小值
    • max指定最大值

  • double(双精度浮点数校验器,要求field的双精度浮点数必须在指定范围内,min指定最小值,max指定最大值)
  • fieldexpression (字段OGNL表达式校验器,要求field满足一个ognl表达式,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过)

  • email(邮件地址校验器,要求如果被校验的属性值非空,则必须是合法的邮件地址)

  • url(网址校验器,要求如果被校验的属性值非空,则必须是合法的url地址)

  • date :日期校验器,要求field的日期值必须在指定范围内
    • min指定最小值
    • max指定最大值

  • 问题:通过配置校验,怎样处理在同一个action中存在多个请求处理方法校验问题?
    • 只需要将校验xml文件名称修改就可以。

      • 原名称:action类名-valication.xml 现在要对action类中某一个方法校验。
      • 修改名称:action类名-action名称-validation.xml

时间: 2024-10-13 01:00:03

Struts2学习第二天——获取参数与数据校验的相关文章

oracle学习 第二章 限制性查询和数据的排序 ——03

这里,我们接着上一小节2.6留下的问题:如果要查询的字符串中含有"_"或"%",又该怎样处理呢? 开始今天的学习. 2.7  如何使用转义(escape)操作符 可以是用个转义(escape)关键字来完成此任务.为了进行练习,我们必须先创建一个临时的表,之后再往该表中插入1行记录,其包含通配符.可能您现在还可能十分不理解例2-13和例2-14的SQL语句.没有问题,您只要照着输入就可以了. 例 2-13 SQL> CREATE TABLE dept_temp

struts2学习(8)struts标签1(数据标签、控制标签)

一.struts2标签简介: struts标签很多,功能强大,这是优点: 但是缺点的话,性能方面可能会,各方面速度啊啥的会降低:有人比较测试,struts性能比jstl低很多: 二.struts2数据标签: com.cy.model.Student.java: package com.cy.model; public class Student { private int id; private String name; private int age; public Student() { s

Struts2学习第二课 Struts2概述

Struts2是一个用来开发MVC应用程序的框架,它提供了Web应用程序开发过程中的一些常见问题飞解决方案: -对来自用户的输入数据进行合法性验证 -统一的布局 -可扩展性 -国际化和本地化 -支持Ajax -表单的重复提交 -文件的上传下载 Struts2和Struts1相比有哪些优势? 在体系结构方面更优秀 -类更少,更高效:在Struts2中无需使用"ActionForm"来封装请求参数 -扩展更容易:Struts2通过拦截器完成了框架的大部分工作,在Struts2中插入一个拦截

【SpringMVC学习06】SpringMVC中的数据校验

这一篇博文主要总结一下springmvc中对数据的校验.在实际中,通常使用较多是前端的校验,比如页面中js校验,对于安全要求较高的建议在服务端也要进行校验.服务端校验可以是在控制层conroller,也可以是在业务层service,controller校验页面请求的参数的合法性,在服务端控制层conroller的校验,不区分客户端类型(浏览器.手机客户端.远程调用):service层主要校验关键业务参数,仅限于service接口中使用的参数.这里主要总结一下何如使用springmvc中contr

struts2学习笔记之十五(输入校验)

输入校验 输入校验和类型转换相辅相成 提交String请求参数,首先是类型转换,再次验证数据的合理性,然后进一步处理 分为客户端校验和服务器校验 客户端:js代码 服务器校验:java代码 客户端校验不能阻止恶意行为,决不能代替服务器校验 客户端也必不可少,由于大部分的浏览行为是正常行为,这样可以降低服务器的负担 两者必不可少. struts2的输入校验完全是模块化的方式 需要检验哪个Action 1.继承ActionSupport类 2.为该Action增加校验规则文件 --struts2会自

Struts2学习(二)———— 表单参数自动封装和参数类型自动转换

前篇文章对struts2的一个入门,重点是对struts2的架构图有一个大概的了解即可,之后的几篇文章,就是细化struts2,将struts2中的各种功能进行梳理,其实学完之后,对struts2的使用不外乎这几点,参数自动封装,拦截器的使用,数据校验,ognl表达(值栈和actionContext的讲解),struts2的标签,struts2的国际化,struts2的文件上传下载. 把这几个功能都学会了使用之后,struts2基本上就学完了.所以接下来的文章就是对这几个功能进行讲解.如何使用.

让Controller支持对平铺参数执行@Valid数据校验

每篇一句 在金字塔塔尖的是实践,学而不思则罔,思而不学则殆(现在很多编程框架都只是教你碎片化的实践) 相关阅读 [小家Java]深入了解数据校验:Java Bean Validation 2.0(JSR303.JSR349.JSR380)Hibernate-Validation 6.x使用案例br/>[[小家Spring]@Validated和@Valid的区别?教你使用它完成Controller参数校验(含级联属性校验)以及原理分析](https://blog.csdn.net/f641385

struts2框架学习笔记4:获取参数

第一种参数获取方式: 编写一个前端页面,提交表单,做示例: <form action="${pageContext.request.contextPath}/Demo1Action"> 用户名:<input type="text" name="name" /><br> 年龄:<input type="text" name="age" /><br>

Struts2学习第三天——保存登陆信息及数据校验

在JSP中服务器端保存数据的作用域通常有request.session.application,它们对应的Servlet API分别是HttpServletRquerst.HttpSession.ServletContext.同样在Struts2的控制器中也需要访问这些作用域. Struts2提供了一个ActionContext类,该类被称为Action上下文或者Action环境,Action可以通过该类访问最常用的ServletAPI. ServletAPI有多种访问方式: 1. Servle