转载自: 开源中国社区 面向对象的15位、18位中国大陆身份证号码解析、验证工具
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.apache.log4j.Logger;
/**
* @ClassName: IDCard
* @Description: 身份证号码,可以解析身份证号码的各个字段,以及验证身份证号码是否有效
* <p>
* 身份证号码构成:6位地址编码+8位生日+3位顺序码+1位校验码
* </p>
* @author
* @company
* @date 2012-12-24
* @version V1.0
*/
public class IDCard {
private static final Logger log = Logger.getLogger(IDCard.class);
/** 身份证号码中的出生日期的格式 */
private static final String BIRTH_DATE_FORMAT = "yyyyMMdd";
/** 身份证的最小出生日期,1900年1月1日 */
private static final Date MIN_BIRTH_DATE = new Date(-2209017600000L);
/** 新版身份证号码长度 */
private static final int NEW_CARD_NUMBER_LENGTH = 18;
/** 旧版身份证号码长度 */
private static final int OLD_CARD_NUMBER_LENGTH = 15;
/** 18位身份证中最后一位校验码 */
private static final char[] VERIFY_CODE = { '1', '0', 'X', '9', '8', '7',
'6', '5', '4', '3', '2' };
/** 18位身份证中,各个数字的生成校验码时的权值 */
private static final int[] VERIFY_CODE_WEIGHT = { 7, 9, 10, 5, 8, 4, 2, 1,
6, 3, 7, 9, 10, 5, 8, 4, 2 };
/** 完整的身份证号码 */
private final String cardNumber;
/** 缓存身份证是否有效,因为验证有效性使用频繁且计算复杂 */
private Boolean cacheValidateResult = null;
/** 缓存出生日期,因为出生日期使用频繁且计算复杂 */
private Date cacheBirthDate = null;
public IDCard(String cardNumber) {
if (null != cardNumber) {
cardNumber = cardNumber.trim();
if (OLD_CARD_NUMBER_LENGTH == cardNumber.length()) {
// 如果是15位身份证号码,则自动转换为18位
cardNumber = contertToNewCardNumber(cardNumber);
}
}
this.cardNumber = cardNumber;
}
/**
* @Title: validate
* @Description: 身份证号码校验
* @return boolean
* @author
* @date 2012-12-24
*/
public boolean validate() {
if (null == this.cacheValidateResult) {
boolean result = true;
try {
// 身份证号码不能为空
result = result && (null != this.cardNumber);
// 身份证号长度是18(新证)
result = result
&& NEW_CARD_NUMBER_LENGTH == this.cardNumber.length();
char ch;
// 身份证号的前17位必须是阿拉伯数字
for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
ch = cardNumber.charAt(i);
result = result && ch >= '0' && ch <= '9';
}
// 身份证号的第18位校验正确
result = result
&& (calculateVerifyCode(cardNumber) == cardNumber
.charAt(NEW_CARD_NUMBER_LENGTH - 1));
// 出生日期不能晚于当前时间,并且不能早于1900年
Date birthDate = getBirthDate();
result = result && null != birthDate;
result = result && birthDate.before(new Date());
result = result && birthDate.after(MIN_BIRTH_DATE);
String birthdayPart = getBirthDayPart();
String realBirthdayPart = this.createBirthDateParser().format(
birthDate);
result = result && (birthdayPart.equals(realBirthdayPart));
} catch (Exception e) {
log.error("Execute method:validate() fail!", e);
result = false;
}
// 完整身份证号码的省市县区检验规则
cacheValidateResult = Boolean.valueOf(result);
}
return cacheValidateResult;
}
/**
* @Title: getAddressCode
* @Description: 获取身份证号码中的地址编码
* @return String
* @author
* @date 2012-12-24
*/
public String getAddressCode() {
checkIfValid();
return this.cardNumber.substring(0, 6);
}
/**
* @Title: getBirthDate
* @Description: 获取身份证号码中的生日
* @return java.util.Date
* @author
* @date 2012-12-24
*/
public Date getBirthDate() {
if (null == this.cacheBirthDate) {
try {
this.cacheBirthDate = createBirthDateParser().parse(
getBirthDayPart());
} catch (ParseException e) {
log.error("Parse BirthDay Part fail!", e);
} catch (Exception e) {
log.error("Parse BirthDay Part fail!", e);
}
}
return new Date(this.cacheBirthDate.getTime());
}
/**
* @Title: isMale
* @Description: 判断是否为男性
* @return boolean
* @author
* @date 2012-12-24
*/
public boolean isMale() {
return 1 == getGenderCode();
}
/**
* @Title: isMale
* @Description: 判断是否为女性
* @return boolean
* @author
* @date 2012-12-24
*/
public boolean isFemal() {
return false == isMale();
}
/**
* @Title: getGenderCode
* @Description: 获取身份证的第17位,奇数为男性,偶数为女性
* @return int
* @author
* @date 2012-12-24
*/
private int getGenderCode() {
checkIfValid();
char genderCode = this.cardNumber.charAt(NEW_CARD_NUMBER_LENGTH - 2);
return (((int) (genderCode - '0')) & 0x1);
}
private String getBirthDayPart() {
return this.cardNumber.substring(6, 14);
}
private SimpleDateFormat createBirthDateParser() {
return new SimpleDateFormat(BIRTH_DATE_FORMAT);
}
private void checkIfValid() {
if (false == validate()) {
throw new RuntimeException("身份证号码不正确!");
}
}
/**
* @Title: calculateVerifyCode
* @Description: 校验码(第十八位数)
* <p>
* 十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0...16 ,先对前17位数字的权求和
* </p>
* <p>
* Ai:表示第i位置上的身份证号码数字值
* </p>
* <p>
* Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
* </p>
* <p>
* 计算模 Y = mod(S, 11)
* </p>
* <p>
* 通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9 8 7 6 5
* 4 3 2
* </p>
* @param cardNumber
* @return char
* @author
* @date 2012-12-24
*/
private static char calculateVerifyCode(CharSequence cardNumber) {
int sum = 0;
char ch;
for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
ch = cardNumber.charAt(i);
sum += ((int) (ch - '0')) * VERIFY_CODE_WEIGHT[i];
}
return VERIFY_CODE[sum % 11];
}
/**
* @Title: contertToNewCardNumber
* @Description: 把15位身份证号码转换到18位身份证号码
* @param oldCardNumber
* 15位身份证号码
* @return:
* @author
* @date 2012-12-24
*/
private static String contertToNewCardNumber(String oldCardNumber) {
/*
* 15位身份证号码与18位身份证号码的区别为:
* 1: 15位身份证号码中,"出生年份"字段是2位,转换时需要补入"19",表示20世纪;
* 2: 15位身份证无最后一位校验码。18位身份证中,校验码根据根据前17位生成
*/
StringBuilder buf = new StringBuilder(NEW_CARD_NUMBER_LENGTH);
buf.append(oldCardNumber.substring(0, 6));
buf.append("19");
buf.append(oldCardNumber.substring(6));
buf.append(IDCard.calculateVerifyCode(buf));
return buf.toString();
}
/**
* @return the cardNumber
*/
public String getCardNumber() {
return cardNumber;
}
}
测试:junit4
注:由于身份证号涉及到个人隐私,故此测试就不写身份证号了,测试的时候,可以写上自己的身份证号.
import java.text.SimpleDateFormat;
import java.util.Date;
import org.junit.Test;
/**
* @ClassName: TestIDCard
* @Description: 测试身份证
* @author
* @company
* @date 2012-12-24
* @version V1.0
*/
public class TestIDCard {
private static final String cardNumber = "xxxxxxxxxxxxxxxxxx";
static IDCard card = new IDCard(cardNumber);
@Test
public void validate() {
System.out.println(card.validate());
}
@Test
public void getAddressCode() {
System.out.println(card.getAddressCode());
}
@Test
public void getBirthDate() {
Date birthDate = card.getBirthDate();
System.out.println("birthDate of Date is:" + birthDate);
if (null != birthDate) {
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd");
System.out.println("birthDate of String is:"
+ sdf.format(birthDate));
}
}
@Test
public void getGender() {
if (card.isMale()) {
System.out.println("男性");
}
if (card.isFemal()) {
System.out.println("女性");
}
}
}
分享到:
相关推荐
18位身份证号码校验位规则 18位身份证号码校验位规则
Oracle身份证号码校验函数,可以将15位身份号码升为18位,检验出生年月等信息
针对身份证号码的校验,以及能自动将身份证从15位升级为18位
做数据录入的最关心身份证信息的合法性,如果是错误的录入也无用,本文件的功能就是做5位和18位公民身份证校验。
一代15位身份证二代18位身份证正则表达式校验,带详细注释,每个字符注释,大厂通用
pb验证身份证号输入的正确性(可以对身份证号码的第18位校验位进行判断) 源码 pb验证身份证号输入的正确性(可以对身份证号码的第18位校验位进行判断) 源码 pb验证身份证号输入的正确性(可以对身份证号码的第18位...
身份证号码18位 和15位 验证的方法加正则表达式
Java校验身份证号码工具类 ---运行几年了,很靠谱。包含15位身份证校验Java校验身份证号码工具类 ---运行几年了,很靠谱。包含15位身份证校验
可以校验15位和18位的身份证号码。 如果输入的18位身份证号码有错,将自动改为正确的号码。
JAVA验证身份证号码校验码是否正确
C++身份证号码校验,身份证号码的排序规则,以及如何用c++对身份证好码进行信息查询
本函数提供18位身份证校验位的计算方法,输入前17位即可得到第18位的校验码!
C#实现身份证号码校验 仅此和大家一起分享 希望大家多多指教
网上很多关于身份证校验的javascript代码,不过都没给全,以下是我添加全了函数一起提供的,希望对大家有所帮助。 javascript身份证号码验证-支持18位带x身份证。
甘肃人口全员管理系统(WIS系统)身份证号码批量校验.
此工具可通过计算,得出身份证号码最后一位。需提供身份证号码前17位,准确率100%。
用C语言写的一个小工具,根据国标算法检验校验身份证号码真伪,只校验身份证号码,不检查信息。
html5进行身份证号码校验,支持15位与18位的身份证号码校验
身份证15位或18位验证
正则表达式验证身份证号码 共两个验证 一个是简单的 只判断是不是15位或者18位, 另外一个方法是复杂的校验,严格判断身份证号码是否合理