import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Date;
import java.util.Objects;
/**
* @author ParanoidCAT
*/
public class IdValidateUtils {
/**
* 省级、直辖市行政区号:
* 11 : 北京 12 : 天津 13 : 河北 14 : 山西 15 : 内蒙古
* 21 : 辽宁 22 : 吉林 23 : 黑龙江
* 31 : 上海 32 : 江苏 33 : 浙江 34 : 安徽 35 : 福建 36 : 江西 37 : 山东
* 41 : 河南 42 : 湖北 43 : 湖南 44 : 广东 45 : 广西 46 : 海南
* 50 : 重庆 51 : 四川 52 : 贵州 53 : 云南 54 : 西藏
* 61 : 陕西 62 : 甘肃 63 : 青海 64 : 宁夏 65 : 新疆
* 71 : 台湾
* 81 : 香港 82 : 澳门
* 91 : 国外
*/
private static final String[] PROVINCE_CODES = {
"11", "12", "13", "14", "15",
"21", "22", "23",
"31", "32", "33", "34", "35", "36", "37",
"41", "42", "43", "44", "45", "46",
"50", "51", "52", "53", "54",
"61", "62", "63", "64", "65",
"71",
"81", "82",
"91"
};
/**
* 15位
*/
private static final int FIFTEEN = 15;
/**
* 6位出生日期格式化模板
*/
private static final String SIX_DIGIT_BIRTH_DATE_PATTERN = "yyMMdd";
/**
* 18位
*/
private static final int EIGHTEEN = 18;
/**
* 8位出生日期格式化模板
*/
private static final String EIGHT_DIGIT_BIRTH_DATE_PATTERN = "yyyyMMdd";
/**
* 加权系数
*/
private static final int[] WEIGHTING_COEFFICIENT = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2};
/**
* 校验码映射
*/
private static final String[] CHECK_CODE_MAPPING = {"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};
private IdValidateUtils() throws Exception {
throw new Exception("no IdValidateUtils instance should be created");
}
/**
* 校验字符串不为null且不为空字符串
*
* @param string 待校验的字符串
* @return true/false
*/
private static boolean isEmpty(String string) {
return string == null || "".equals(string);
}
/**
* 校验身份证号是否符合规则
*
* @param idNumberString 待校验的身份证号
* @return true/false
*/
public static boolean isIdNumberValidated(String idNumberString) {
if (isEmpty(idNumberString)) {
return false;
}
if (idNumberString.length() == FIFTEEN) {
return isFifteenDigitIdNumberValidated(idNumberString);
}
if (idNumberString.length() == EIGHTEEN) {
return isEighteenDigitIdNumberValidated(idNumberString);
}
return false;
}
/**
* 校验15位身份证号
* 15位身份证号的构成如下:
* 省级、直辖市行政区号2位
* 地、市级行政区号2位
* 区、县级行政区号2位
* 出生年2位
* 出生月2位
* 出生日2位
* 序列号3位,其中奇数分配给男性,偶数分配给女性
*
* @param idNumberString 待校验的身份证号
* @return true/false
*/
private static boolean isFifteenDigitIdNumberValidated(String idNumberString) {
if (isDigital(idNumberString)) {
if (startsWithProvinceCode(idNumberString)) {
String birthDateString = idNumberString.substring(6, 12);
return isBirthDateValidated(birthDateString, SIX_DIGIT_BIRTH_DATE_PATTERN);
}
}
return false;
}
/**
* 校验18位身份证号
* 18位身份证号的构成如下:
* 省级、直辖市行政区号2位
* 地、市级行政区号2位
* 区、县级行政区号2位
* 出生年4位
* 出生月2位
* 出生日2位
* 序列号3位,其中奇数分配给男性,偶数分配给女性
* 校验码1位,由前17位号码每位乘以其对应索引位置的加权系数后求和,再根据和除以11的余数作为索引取校验码映射中对应索引的校验码
* 可参考类内私有常量["加权系数","校验码映射"]
*
* @param idNumberString 待校验的身份证号
* @return true/false
*/
private static boolean isEighteenDigitIdNumberValidated(String idNumberString) {
String topSeventeenDigit = idNumberString.substring(0, 17);
if (isDigital(topSeventeenDigit)) {
if (startsWithProvinceCode(idNumberString)) {
String birthDateString = idNumberString.substring(6, 14);
if (isBirthDateValidated(birthDateString, EIGHT_DIGIT_BIRTH_DATE_PATTERN)) {
String checkCode = getCheckCodeByWeightingSum(getWeightingSum(topSeventeenDigit.toCharArray()));
return Objects.equals(idNumberString.substring(17, 18), checkCode);
}
}
}
return false;
}
/**
* 校验字符串是否全部由数字构成
*
* @param string 待校验的字符串
* @return true/false
*/
private static boolean isDigital(String string) {
char[] chars = string.toCharArray();
for (char c : chars) {
if (!Character.isDigit(c)) {
return false;
}
}
return true;
}
/**
* 校验字符串是否以省、直辖市代码开头
*
* @param string 待校验的字符串
* @return true/false
*/
private static boolean startsWithProvinceCode(String string) {
String provinceCode = string.substring(0, 2);
return Arrays.asList(PROVINCE_CODES).contains(provinceCode);
}
/**
* 校验字符串是否为合法的日期格式
*
* @param string 待校验的字符串
* @return true/false
*/
private static boolean isBirthDateValidated(String string, String pattern) {
DateFormat dateFormat = new SimpleDateFormat(pattern);
try {
Date birthDate = dateFormat.parse(string);
String temp = dateFormat.format(birthDate);
return Objects.equals(temp, string);
} catch (ParseException e) {
return false;
}
}
/**
* 获取加权和
*
* @param chars 待计算的字节数组
* @return 加权和
*/
private static int getWeightingSum(char[] chars) {
int weightingSum = 0;
for (int i = 0; i < chars.length; i++) {
weightingSum += WEIGHTING_COEFFICIENT[i] * Integer.parseInt(String.valueOf(chars[i]));
}
return weightingSum;
}
/**
* 根据加权和获取校验码
*
* @param weightingSum 加权和
* @return 校验码
*/
private static String getCheckCodeByWeightingSum(int weightingSum) {
return CHECK_CODE_MAPPING[weightingSum % 11];
}
/**
* 将十五位身份证号码转换为十八位
*
* @param idNumber 十五位身份证号码
* @return 十八位身份证号码
*/
public static String convertFifteenDigitIdCardToEighteenDigitIdCard(String idNumber) {
if (isEmpty(idNumber)) {
throw new IllegalArgumentException("身份证字符不可为空");
}
if (idNumber.length() != FIFTEEN) {
throw new IllegalArgumentException("身份证字符的长度错误,无法进行转换");
}
if (!isFifteenDigitIdNumberValidated(idNumber)) {
throw new IllegalArgumentException("身份证校验失败,无法进行转换");
}
if (startsWithProvinceCode(idNumber)) {
String birthDateString = idNumber.substring(6, 12);
if (isBirthDateValidated(birthDateString, SIX_DIGIT_BIRTH_DATE_PATTERN)) {
String topSeventeenDigit = idNumber.substring(0, 6) + getPossibleYear(idNumber) + idNumber.substring(6, 15);
return topSeventeenDigit + getCheckCodeByWeightingSum(getWeightingSum(topSeventeenDigit.toCharArray()));
}
}
return null;
}
/**
* 根据当前日期判断可能的年限
* 我国(中华人民共和国)自2004年1月1日起开始换发第二代身份证
* 根据我国人民在20世纪的平均寿命判断:第一代身份证的身份证号的出生年为为00,01,02,03的居民为2000年后出生
* 剩余一代身份证居民为1900年后出生
*
* @param idNumberString 身份证字符串
* @return 可能的年限
*/
private static String getPossibleYear(String idNumberString) {
int year = Integer.parseInt(idNumberString.substring(7, 9));
if (year >= 0 && year < 4) {
return "20";
} else {
return "19";
}
}
}