Struts(二十五):自定义验证器
-
编程验证
Struts2提供了一个Validateable接口,可以使用Action类实现这个接口以提供编程验证;
ActionSupport类已经实现了Validateable接口。
public class MyAction extends ActionSupport{ private String age; public void setAge(String age) { this.age = age; } public String getAge() { return age; } @Override public void validate() { if (age == null || age.equals("")) { addFieldError("age", getText("age.null")); } super.validate(); } }
-
如何定义验证器?
1)自定义验证器必须实现Validator接口;
2)ValidatorSupport和FieldValidatorSupport实现了Validator接口;
如需要实现普通的验证程序,可以继承ValidatorSupport类;
如果要实现字段的验证程序,可以继承FieldValidatorSupport类;
如果验证器需要接受一个输入参数,需要为这个参数增加一个相应的属性。
3)注册验证器:自定义验证器需要在类路径里的某个validators.xml文件里注册,验证框架首先在根目录下找validators.xml文件,如果没有找到validators.xml文件,验证框架将调用默认的验证设置,即xwork-core.jar下com.opensymphony.xwork2.validator.validators包下default.xml文件里的配置信息。
-
以身份证验证器为例,展示如何定义验证器。
1、自定义filed验证器com.dx.strtuts2.validators.IDCardValidator.java
package com.dx.strtuts2.validators; import java.text.ParseException; import com.opensymphony.xwork2.validator.ValidationException; import com.opensymphony.xwork2.validator.validators.FieldValidatorSupport; public class IDCardValidator extends FieldValidatorSupport { @Override public void validate(Object object) throws ValidationException { // 1、获取请求字段名称及值 String fieldName = getFieldName(); Object fieldValue = getFieldValue(fieldName, object); // 2、验证参数值是否合法 IDCardValidate validate = new IDCardValidate(); Boolean result = false; try { result = validate.validate((String) fieldValue); } catch (ParseException e) { result = false; e.printStackTrace(); } // 3、如果验证事变,返回失败信息 if (!result) { addFieldError(fieldName, object); } } }
com.dx.strtuts2.validators.IDCardValidate.java
1 package com.dx.strtuts2.validators; 2 3 import java.text.ParseException; 4 import java.text.SimpleDateFormat; 5 import java.util.GregorianCalendar; 6 import java.util.Hashtable; 7 import java.util.regex.Matcher; 8 import java.util.regex.Pattern; 9 import java.util.Calendar; 10 11 public class IDCardValidate { 12 /** 13 * 1、号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码, 14 * 三位数字顺序码和一位数字校验码。 15 * 16 * 2、地址码(前六位数) 表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。 17 * 18 * 3、出生日期码(第七位至十四位) 表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。 19 * 20 * 4、顺序码(第十五位至十七位) 21 * 表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号,顺序码的奇数分配给男性,偶数分配给女性。 22 * 23 * 5、校验码(第十八位数) (1)十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, ... , 16 24 * ,先对前17位数字的权求和 Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 25 * 3 7 9 10 5 8 4 2 (2)计算模 Y = mod(S, 11) (3)通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 26 * 8 9 10 校验码: 1 0 X 9 8 7 6 5 4 3 2 27 * 28 * 所以我们就可以大致写一个函数来校验是否正确了。 29 */ 30 31 /** 32 * 功能:身份证的有效验证 33 * 34 * @param IDStr 35 * 身份证号 36 * @return 有效:true 无效:false 37 * @throws ParseException 38 */ 39 public boolean validate(String IDStr) throws ParseException { 40 String errorInfo = "";// 记录错误信息 41 String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4", "3", "2" }; 42 String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2" }; 43 // String[] Checker = {"1","9","8","7","6","5","4","3","2","1","1"}; 44 String Ai = ""; 45 46 // ================ 号码的长度 15位或18位 ================ 47 if (IDStr.length() != 15 && IDStr.length() != 18) { 48 errorInfo = "号码长度应该为15位或18位。"; 49 System.out.println(errorInfo); 50 return false; 51 } 52 // =======================(end)======================== 53 54 // ================ 数字 除最后以为都为数字 ================ 55 if (IDStr.length() == 18) { 56 Ai = IDStr.substring(0, 17); 57 } else if (IDStr.length() == 15) { 58 Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15); 59 } 60 if (isNumeric(Ai) == false) { 61 errorInfo = "15位号码都应为数字 ; 18位号码除最后一位外,都应为数字。"; 62 System.out.println(errorInfo); 63 return false; 64 } 65 // =======================(end)======================== 66 67 // ================ 出生年月是否有效 ================ 68 String strYear = Ai.substring(6, 10);// 年份 69 String strMonth = Ai.substring(10, 12);// 月份 70 String strDay = Ai.substring(12, 14);// 月份 71 72 if (this.isDate(strYear + "-" + strMonth + "-" + strDay) == false) { 73 errorInfo = "生日无效。"; 74 System.out.println(errorInfo); 75 return false; 76 } 77 78 GregorianCalendar gc = new GregorianCalendar(); 79 SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd"); 80 if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150 81 || (gc.getTime().getTime() - s.parse(strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) { 82 errorInfo = "生日不在有效范围。"; 83 System.out.println(errorInfo); 84 return false; 85 } 86 if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) { 87 errorInfo = "月份无效"; 88 System.out.println(errorInfo); 89 return false; 90 } 91 if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) { 92 errorInfo = "日期无效"; 93 System.out.println(errorInfo); 94 return false; 95 } 96 // =====================(end)===================== 97 98 // ================ 地区码时候有效 ================ 99 Hashtable<String, String> h = GetAreaCode(); 100 if (h.get(Ai.substring(0, 2)) == null) { 101 errorInfo = "地区编码错误。"; 102 System.out.println(errorInfo); 103 return false; 104 } 105 // ============================================== 106 107 // ================ 判断最后一位的值 ================ 108 int TotalmulAiWi = 0; 109 for (int i = 0; i < 17; i++) { 110 TotalmulAiWi = TotalmulAiWi + Integer.parseInt(String.valueOf(Ai.charAt(i))) * Integer.parseInt(Wi[i]); 111 } 112 int modValue = TotalmulAiWi % 11; 113 String strVerifyCode = ValCodeArr[modValue]; 114 Ai = Ai + strVerifyCode; 115 116 if (IDStr.length() == 18) { 117 if (Ai.equals(IDStr) == false) { 118 errorInfo = "身份证无效,最后一位错误"; 119 System.out.println(errorInfo); 120 return false; 121 } 122 } else { 123 System.out.println("所在地区:" + h.get(Ai.substring(0, 2).toString())); 124 System.out.println("新身份证号:" + Ai); 125 return true; 126 } 127 // =====================(end)===================== 128 System.out.println("所在地区:" + h.get(Ai.substring(0, 2).toString())); 129 return true; 130 } 131 132 /** 133 * 功能:设置地区编码 134 * 135 * @return Hashtable 对象 136 */ 137 private Hashtable<String, String> GetAreaCode() { 138 Hashtable<String, String> hashtable = new Hashtable<String, String>(); 139 hashtable.put("11", "北京"); 140 hashtable.put("12", "天津"); 141 hashtable.put("13", "河北"); 142 hashtable.put("14", "山西"); 143 hashtable.put("15", "内蒙古"); 144 hashtable.put("21", "辽宁"); 145 hashtable.put("22", "吉林"); 146 hashtable.put("23", "黑龙江"); 147 hashtable.put("31", "上海"); 148 hashtable.put("32", "江苏"); 149 hashtable.put("33", "浙江"); 150 hashtable.put("34", "安徽"); 151 hashtable.put("35", "福建"); 152 hashtable.put("36", "江西"); 153 hashtable.put("37", "山东"); 154 hashtable.put("41", "河南"); 155 hashtable.put("42", "湖北"); 156 hashtable.put("43", "湖南"); 157 hashtable.put("44", "广东"); 158 hashtable.put("45", "广西"); 159 hashtable.put("46", "海南"); 160 hashtable.put("50", "重庆"); 161 hashtable.put("51", "四川"); 162 hashtable.put("52", "贵州"); 163 hashtable.put("53", "云南"); 164 hashtable.put("54", "西藏"); 165 hashtable.put("61", "陕西"); 166 hashtable.put("62", "甘肃"); 167 hashtable.put("63", "青海"); 168 hashtable.put("64", "宁夏"); 169 hashtable.put("65", "新疆"); 170 hashtable.put("71", "台湾"); 171 hashtable.put("81", "香港"); 172 hashtable.put("82", "澳门"); 173 hashtable.put("91", "国外"); 174 return hashtable; 175 } 176 177 /** 178 * 功能:判断字符串是否为数字 179 * 180 * @param str 181 * @return 182 */ 183 private boolean isNumeric(String str) { 184 Pattern pattern = Pattern.compile("[0-9]*"); 185 Matcher isNum = pattern.matcher(str); 186 if (isNum.matches()) { 187 return true; 188 } else { 189 return false; 190 } 191 } 192 193 /** 194 * 功能:判断字符串是否为日期格式 195 * 196 * @param str 197 * @return 198 */ 199 public boolean isDate(String strDate) { 200 Pattern pattern = Pattern.compile( 201 "^((\\d{2}(([02468][048])|([13579][26]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])))))|(\\d{2}(([02468][1235679])|([13579][01345789]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|(1[0-9])|(2[0-8]))))))(\\s(((0?[0-9])|([1-2][0-3]))\\:([0-5]?[0-9])((\\s)|(\\:([0-5]?[0-9])))))?$"); 202 Matcher m = pattern.matcher(strDate); 203 if (m.matches()) { 204 return true; 205 } else { 206 return false; 207 } 208 } 209 }
2、修改struts.xml
<action name="myIdCardValidation" class="com.dx.struts2.myvalidations.MyValidationAction" method="idcardExecute"> <result>/success.jsp</result> <result name="input">/index.jsp</result> </action>
3、在src根目录下创建validators.xml,并注册com.dx.strtuts2.validators.IDCardValidator.java到validators.xml中(注册规则参考:xwork-core.jar下com.opensymphony.xwork2.validator.validators包下default.xml文件里的配置信息)
1 <?xml version="1.0" encoding="UTF-8"?> 2 <!DOCTYPE validators PUBLIC 3 "-//Apache Struts//XWork Validator Definition 1.0//EN" 4 "http://struts.apache.org/dtds/xwork-validator-definition-1.0.dtd"> 5 6 <!-- START SNIPPET: validators-default --> 7 <validators> 8 <validator name="idcard" 9 class="com.dx.strtuts2.validators.IDCardValidator" /> 10 </validators>
4、添加Action com.dx.struts2.myvalidations.MyValidationAction.java
package com.dx.struts2.myvalidations; import com.opensymphony.xwork2.ActionSupport; public class MyValidationAction extends ActionSupport { private static final long serialVersionUID = 1L; private String idCard; public String getIdCard() { return idCard; } public void setIdCard(String idCard) { this.idCard = idCard; } public String idcardExecute() { System.out.println("id card execute..."); return SUCCESS; } }
5、在包com.dx.struts2.myvalidations下添加验证规则文件:MyValidationAction-myIdCardValidation-validation.xml
<!DOCTYPE validators PUBLIC "-//Apache Struts//XWork Validator 1.0.2//EN" "http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd"> <validators> <field name="idCard"> <field-validator type="idcard"> <message>error id card!!!</message> </field-validator> </field> </validators>
6、添加index.jsp页面:
<s:debug></s:debug> <s:actionerror /> <s:form action="myIdCardValidation" method="post"> <s:textfield name="idCard" label="IdCard"></s:textfield> <s:submit label="Submit"></s:submit> </s:form>
7、验证输入一个错误id card,提交后页面信息
输入一个正确的id card,成功跳转到了success页面。
基础才是编程人员应该深入研究的问题,比如:
1)List/Set/Map内部组成原理|区别
2)mysql索引存储结构&如何调优/b-tree特点、计算复杂度及影响复杂度的因素。。。
3)JVM运行组成与原理及调优
4)Java类加载器运行原理
5)Java中GC过程原理|使用的回收算法原理
6)Redis中hash一致性实现及与hash其他区别
7)Java多线程、线程池开发、管理Lock与Synchroined区别
8)Spring IOC/AOP 原理;加载过程的。。。
【+加关注】。