转载自: 开源中国社区 面向对象的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("女性");
}
}
}