1、设置接口

@Constraint(validatedBy = ChineseIdNumberValidator.class)
@Target({ElementType.METHOD,
ElementType.FIELD,
ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface IdNumber { String message() default "{common.IdNumber.Pattern}"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {};
}

2、编写Validator

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger; import javax.validation.ConstraintValidatorContext; /**
*
* <p>
* 判断18位身份证的合法性
* </p>
* 根据〖中华人民共和国国家标准GB11643-1999〗中有关公民身份号码的规定,公民身份号码是特征组合码,由十七位数字本体码和一位数字校验码组成。
* 排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。
* <p>
* 顺序码: 表示在同一地址码所标识的区域范围内,对同年、同月、同 日出生的人编定的顺序号,顺序码的奇数分配给男性,偶数分配 给女性。
* </p>
* <p>
* 1.前1、2位数字表示:所在省份的代码; 2.第3、4位数字表示:所在城市的代码; 3.第5、6位数字表示:所在区县的代码;
* 4.第7~14位数字表示:出生年、月、日; 5.第15、16位数字表示:所在地的派出所的代码; 6.第17位数字表示性别:奇数表示男性,偶数表示女性;
* 7.第18位数字是校检码:也有的说是个人信息码,一般是随计算机的随机产生,用来检验身份证的正确性。校检码可以是0~9的数字,有时也用x表示。
* </p>
* <p>
* 第十八位数字(校验码)的计算方法为: 1.将前面的身份证号码17位数分别乘以不同的系数。从第一位到第十七位的系数分别为:7 9 10 5 8 4 2 1
* 6 3 7 9 10 5 8 4 2
* </p>
* <p>
* 2.将这17位数字和系数相乘的结果相加。
* </p>
* <p>
* 3.用加出来和除以11,看余数是多少?
* </p>
* 4.余数只可能有0 1 2 3 4 5 6 7 8 9 10这11个数字。其分别对应的最后一位身份证的号码为1 0 X 9 8 7 6 5 4 3 2。
* <p>
* 5.通过上面得知如果余数是2,就会在身份证的第18位数字上出现罗马数字的Ⅹ。如果余数是10,身份证的最后一位号码就是2。
* </p>
*
* <p>
* 对于15位身份证,不需要判断最后一位验证码,最后一位代表性别,奇数表示男性,偶数表示女性 默认使用18位身份证号,15位也支持
* </p>
*
* @author rooseek
*
*
*/
public class ChineseIdNumberValidator implements ConstraintValidator { private final static Map<String, String> code2Province = new HashMap<String, String>() { private static final long serialVersionUID = 1L; {
this.put("11", "北京");
this.put("12", "天津");
this.put("13", "河北");
this.put("14", "山西");
this.put("15", "内蒙古");
this.put("21", "辽宁");
this.put("22", "吉林");
this.put("23", "黑龙江");
this.put("31", "上海");
this.put("32", "江苏");
this.put("33", "浙江");
this.put("34", "安徽");
this.put("35", "福建");
this.put("36", "江西");
this.put("37", "山东");
this.put("41", "河南");
this.put("42", "湖北");
this.put("43", "湖南");
this.put("44", "广东");
this.put("45", "广西");
this.put("46", "海南");
this.put("50", "重庆");
this.put("51", "四川");
this.put("52", "贵州");
this.put("53", "云南");
this.put("54", "西藏");
this.put("61", "陕西");
this.put("62", "甘肃");
this.put("63", "青海");
this.put("64", "宁夏");
this.put("65", "新疆");
}
}; private static final int weightFactor[] = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2}; private static final String checkCode[] = {"1", "0", "X", "9", "8", "7", "6", "5",
"4", "3", "2"}; private static final GregorianCalendar calendar = (GregorianCalendar) GregorianCalendar.getInstance(); @Override
public void initialize(IdNumber constraintAnnotation) {
} @Override
public boolean isValid(String idNumber, ConstraintValidatorContext context) {
if (idNumber == null) {
return true;
} return isValid(idNumber);
} protected boolean isValid(String idNumber) {
if (idNumber == null) {
return true;
} if (idNumber.length() == 15) {
return isValid15(idNumber);
} else if (idNumber.length() == 18) {
return isValid18(idNumber);
} return false;
} private boolean isValid15(String idNumber) {
if (idNumber == null || idNumber.length() != 15) {
return false;
} if (!isDigital(idNumber)) {
return false;
} //check province code
String province = idNumber.substring(0, 2);
if (!code2Province.containsKey(province)) {
return false;
} //check birthday
String birthday = idNumber.substring(6, 12);
int month = Integer.parseInt(idNumber.substring(8, 10));
int day = Integer.parseInt(idNumber.substring(10, 12)); //check birthday format and whether this date is before current time
Date birthDate = null;
try {
birthDate = new SimpleDateFormat("yyMMdd").parse(birthday);
} catch (ParseException ex) {
Logger.getLogger(ChineseIdNumberValidator.class.getName()).log(Level.SEVERE, null, ex);
}
if (birthDate == null || System.currentTimeMillis() <= birthDate.getTime()) {
return false;
} //check year, month, day
calendar.setTime(birthDate);
if (calendar.get(Calendar.YEAR) < IdNumberConstant.MIN_YEAR) {
return false;
} if (month < 1 || month > 12) {
return false;
}
switch (month) {
case 1:
case 3:
case 5:
case 7:
case 8:
case 10:
case 12:
if (day < 1 || day > 31) {
return false;
}
case 4:
case 6:
case 9:
case 11:
if (day < 1 || day > 30) {
return false;
}
case 2:
if (calendar.isLeapYear(calendar.get(Calendar.YEAR))) {
if (day < 1 || day > 29) {
return false;
}
} else {
if (day < 1 || day > 28) {
return false;
}
}
} return true;
} private boolean isValid18(String idNumber) {
if (idNumber == null || idNumber.length() != 18) {
return false;
} String idNumber17 = idNumber.substring(0, 17);
String idNumber18Code = idNumber.substring(17, 18);
char charArray[]; //check prefix 17 numbers are all digital
if (isDigital(idNumber17)) {
charArray = idNumber17.toCharArray();
} else {
return false;
} //check region code
String province = idNumber.substring(0, 2);
if (!code2Province.containsKey(province)) {
return false;
} //check birthday
String birthday = idNumber.substring(6, 14);
int year = Integer.parseInt(idNumber.substring(6, 10));
int month = Integer.parseInt(idNumber.substring(10, 12));
int day = Integer.parseInt(idNumber.substring(12, 14)); //check birthday format and whether this date is before current time
Date birthDate = null;
try {
birthDate = new SimpleDateFormat("yyyyMMdd").parse(birthday);
} catch (ParseException ex) {
Logger.getLogger(ChineseIdNumberValidator.class.getName()).log(Level.SEVERE, null, ex);
}
if (birthDate == null || System.currentTimeMillis() <= birthDate.getTime()) {
return false;
} //check year, month, day
if (year < IdNumberConstant.MIN_YEAR) {
return false;
} if (month < 1 || month > 12) {
return false;
}
switch (month) {
case 1:
case 3:
case 5:
case 7:
case 8:
case 10:
case 12:
if (day < 1 || day > 31) {
return false;
}
break;
case 4:
case 6:
case 9:
case 11:
if (day < 1 || day > 30) {
return false;
}
break;
case 2:
if (isLeapYear(year)) {
if (day < 1 || day > 29) {
return false;
}
} else {
if (day < 1 || day > 28) {
return false;
}
}
break;
} //check the sum of prefix 17 numbers mutilplied by weighted factor equals the last check code
if (null != charArray) {
if (!idNumber18Code.equals(
getCheckCode(
getPowerSum(
char2Int(charArray))))) {
return false;
} } return true;
} @SuppressWarnings("unused")
private String convert15To18(String idNumber) {
if (idNumber.length() != 15) {
return null;
} String idNumber17;
if (isDigital(idNumber)) {
String birthday = idNumber.substring(6, 12);
Date birthdate = null;
try {
birthdate = new SimpleDateFormat("yyMMdd").parse(birthday);
} catch (ParseException ex) {
Logger.getLogger(ChineseIdNumberValidator.class.getName()).log(Level.SEVERE, null, ex);
} calendar.setTime(birthdate);
String year = String.valueOf(calendar.get(Calendar.YEAR));
idNumber17 = idNumber.substring(0, 6) + year + idNumber.substring(8);
char charArray[] = idNumber17.toCharArray();
if (null != charArray) {
int sum17 = getPowerSum(char2Int(charArray));
idNumber17 += getCheckCode(sum17);
}
} else {
return null;
}
return idNumber17;
} private boolean isDigital(String str) {
return str == null || "".equals(str) ? false : str.matches("[0-9]*");
} private int getPowerSum(int[] bit) {
int sum = 0;
if (weightFactor.length != bit.length) {
return sum;
}
for (int i = 0; i < bit.length; i++) {
sum = sum + bit[i] * weightFactor[i];
}
return sum;
} private String getCheckCode(int sum17) {
return checkCode[sum17 % 11];
} private int[] char2Int(char[] charArray) throws NumberFormatException {
int[] array = new int[charArray.length];
int k = 0;
for (char temp : charArray) {
array[k++] = Integer.parseInt(String.valueOf(temp));
}
return array;
} @SuppressWarnings("unused")
private boolean isValidIdRegion(String region) {
if (!Regions.isValidRegion(region)) {
return false;
} //idNumber start with something like "110000 北京市/110100 市辖区/110200 县" is not valid
if (region.substring(4, 6).equals(IdNumberConstant.CITY_SUFFIX)) {
return false;
} //台湾 香港 澳门暂时屏蔽
String province = region.substring(0, 2);
if (!isValidProvince(province)) {
return false;
} return true;
} private boolean isValidProvince(String province) {
return !(province.equals(IdNumberConstant.TAIWANG)
|| province.equals(IdNumberConstant.HONGKONG)
|| province.equals(IdNumberConstant.MACAU));
} private boolean isLeapYear(int year) {
return (year % 4 == 0) && (year % 100 != 0) || (year % 400 == 0);
}
}

大陆居民身份证验证方法(java)的更多相关文章

  1. 18位身份证验证(Java)加入身份证输入验证是否满足18位代码(修订稿)

    package day20181016; /** * 身份证的验证 34052419800101001X * */ import java.util.Scanner; public class Zuo ...

  2. 18位身份证验证(Java)

    我的代码: package day20181016;/** * 身份证的验证 34052419800101001X * */import java.util.Scanner;public class ...

  3. Java连接MySQL数据库实现用户名密码的验证方法 Java语句中sql查询语句'' ""作用

    //方法一,可以验证登录,但方法不实用.package com.swift; import java.sql.Connection; import java.sql.DriverManager; im ...

  4. 【NumberValidators】大陆身份证验证

    需要说明的是这里的大陆身份证识别并不是公安局联网的识别,而是按国标GB 11643进行的验证,所以其验证结果只能说符合国标规范,但不能保证该身份证一定真实存在,如果你实际需求是希望身份证一定真实存在, ...

  5. java身份证验证

    import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; impor ...

  6. java对身份证验证及正则表达式解析

    原文地址:http://www.cnblogs.com/zhongshengzhen/ java对身份证验证及正则表达式解析 package service; import java.text.Par ...

  7. java 常用的验证方法帮助类

    import java.text.ParseException; import java.util.Collection; import java.util.Map; /** * 常用的验证方法帮助类 ...

  8. Java国际化号码验证方法,国内手机号正则表达式

    Java国际化号码验证方法,国内手机号正则表达式 中国电信号段 133.149.153.173.177.180.181.189.199 中国联通号段 130.131.132.145.155.156.1 ...

  9. Java基础之身份证验证

    //简约版package test; import java.util.Scanner; public class ID { /** * 匹配算法 : 1) 得到17位身份证号码与下面给出的17位 2 ...

随机推荐

  1. Hbase 建表基本命令总结

    访问hbase,以及操作hbase,命令不用使用分号 hbase shell 进入hbase list 查看表 hbase shell -d hbase(main):024:0> scan '. ...

  2. SMG12232ZK标准字符点阵型液晶显示模块的演示程序[C51编程语言][MCS51并行接口方式]

    //SMG12232ZK标准字符点阵型液晶显示模块的演示程序[C51编程语言][MCS51并行接口方式] //应用产品: SMG12232ZK标准中文字符点阵型液晶显示模块 // 本演示程序适用于SM ...

  3. 给定金额m和红包数量n

    这一题如果是采用暴力手段,不一定能获得正确答案,而且也非常耗时. 所以下面我们采用一个小技巧,也就是先产生n-1个红包,总得sum<m的,这样最后只要添加一个sum-m的红包钱数就可以了. 具体 ...

  4. python打包成exe(py2exe)

    对比了几个打包工具,发现py2exe更好用一点,一个命令就可以. 1.获取 http://prdownloads.sourceforge.net/py2exe 下载python版本对应的.直接下载然后 ...

  5. mysql日期格式说明符

  6. Esper系列(十四)Contained-Event Selection

    功能:该语法是针对所查询事件中的属性又是另一种属性的查询结果控制. 格式: 1  "+j); 19      bean.setBean(item); 20      list.add(bea ...

  7. SQL2008-删除时间字段重复的方法

    ID  EMAgitation_ID    YieldDateTime 1   2                              2012-02-11 10:18:54.0002   2  ...

  8. 一种从JSON数据创建Java类的高效办法

    <一种从JSON数据创建Java类的高效办法> 作者:chszs,转载需注明.博客主页:http://blog.csdn.net/chszs JSON格式的数据经常会遇到,比如调用Web服 ...

  9. 发布maven的web项目,tomcat显示已启动,但是访问出错

    sts中如果是maven的web工程,一定要先进行maven install,再发布工程,否则不会出任何错误,tomcat还访问不了

  10. Windows下环境变量配置

    JAVA_HOME=C:\Program Files\Java\jdk1.6.0_33   PATH+=%JAVA_HOME%\bin;   CLASSPATH=.;%JAVA_HOME%\lib\d ...