身份证号码校验

package com.haier.hmms.util;

import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * 验证身份证号码 身份证号码, 可以解析身份证号码的各个字段,以及验证身份证号码是否有效; 身份证号码构成:6位地址编码+8位生日+3位顺序码+1位校验码
 * 
 * @ClassName: CheckIdCard
 * @Description: TODO
 * @author suncy
 * @date 2013-1-4 上午11:06:09
 * 
 */
public class CheckIdCard {

 private String cardNumber; // 完整的身份证号码
 private Boolean cacheValidateResult = null; // 缓存身份证是否有效,因为验证有效性使用频繁且计算复杂
 private Date cacheBirthDate = null; // 缓存出生日期,因为出生日期使用频繁且计算复杂
 private final static String BIRTH_DATE_FORMAT = "yyyyMMdd"; // 身份证号码中的出生日期的格式
 private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L); // 身份证的最小出生日期,1900年1月1日
 private final static int NEW_CARD_NUMBER_LENGTH = 18;
 private final static int OLD_CARD_NUMBER_LENGTH = 15;
 private final static char[] VERIFY_CODE = { '1', '0', 'X', '9', '8', '7',
   '6', '5', '4', '3', '2' }; // 18位身份证中最后一位校验码
 private final static int[] VERIFY_CODE_WEIGHT = { 7, 9, 10, 5, 8, 4, 2, 1,
   6, 3, 7, 9, 10, 5, 8, 4, 2 };// 18位身份证中,各个数字的生成校验码时的权值

 public boolean validate() {
  if (null == cacheValidateResult) {
   boolean result = true;
   result = result && (null != cardNumber); // 身份证号不能为空
   result = result && NEW_CARD_NUMBER_LENGTH == cardNumber.length(); // 身份证号长度是18(新证)
   // 身份证号的前17位必须是阿拉伯数字
   for (int i = 0; result && i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
    char ch = cardNumber.charAt(i);
    result = result && ch >= '0' && ch <= '9';
   }
   // 身份证号的第18位校验正确
   result = result
     && (calculateVerifyCode(cardNumber) == cardNumber
       .charAt(NEW_CARD_NUMBER_LENGTH - 1));
   // 出生日期不能晚于当前时间,并且不能早于1900年
   try {
    Date birthDate = this.getBirthDate();
    result = result && null != birthDate;
    result = result && birthDate.before(new Date());
    result = result && birthDate.after(MINIMAL_BIRTH_DATE);
    /**
     * 出生日期中的年、月、日必须正确,比如月份范围是[1,12],日期范围是[1,31],还需要校验闰年、大月、小月的情况时,
     * 月份和日期相符合
     */
    String birthdayPart = this.getBirthDayPart();
    String realBirthdayPart = this.createBirthDateParser().format(
      birthDate);
    result = result && (birthdayPart.equals(realBirthdayPart));
   } catch (Exception e) {
    result = false;
   }
   cacheValidateResult = Boolean.valueOf(result);// TODO
               // 完整身份证号码的省市县区检验规则
  }
  return cacheValidateResult;
 }

 /**
  * 如果是15位身份证号码,则自动转换为18位
  * 
  * @param cardNumber
  * @return 
  */
 public CheckIdCard(String cardNumber) {
  if (null != cardNumber) {
   cardNumber = cardNumber.trim();
   if (OLD_CARD_NUMBER_LENGTH == cardNumber.length()) {
    cardNumber = contertToNewCardNumber(cardNumber);
   }
  }
  this.cardNumber = cardNumber;
 }

 public String getCardNumber() {
  return cardNumber;
 }

 public String getAddressCode() {
  this.checkIfValid();
  return this.cardNumber.substring(0, 6);
 }

 public Date getBirthDate() {
  if (null == this.cacheBirthDate) {
   try {
    this.cacheBirthDate = this.createBirthDateParser().parse(
      this.getBirthDayPart());
   } catch (Exception e) {
    throw new RuntimeException("身份证的出生日期无效");
   }
  }
  return new Date(this.cacheBirthDate.getTime());
 }

 public boolean isMale() {
  return 1 == this.getGenderCode();
 }

 public boolean isFemal() {
  return false == this.isMale();
 }

 /**
  * 获取身份证的第17位,奇数为男性,偶数为女性
  * 
  * @return
  */
 private int getGenderCode() {
  this.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 == this.validate()) {
   throw new RuntimeException("身份证号码不正确!");
  }
 }

 /**
  * 校验码(第十八位数):
  * 
  * 十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0...16 ,先对前17位数字的权求和;
  * Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
  * 2; 计算模 Y = mod(S, 11)< 通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9
  * 8 7 6 5 4 3 2
  * 
  * @param cardNumber
  * @return
  */
 private static char calculateVerifyCode(CharSequence cardNumber) {
  int sum = 0;
  for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
   char ch = cardNumber.charAt(i);
   sum += ((int) (ch - '0')) * VERIFY_CODE_WEIGHT[i];
  }
  return VERIFY_CODE[sum % 11];
 }

 /**
  * 把15位身份证号码转换到18位身份证号码<br>
  * 15位身份证号码与18位身份证号码的区别为:<br>
  * 1、15位身份证号码中,"出生年份"字段是2位,转换时需要补入"19",表示20世纪<br>
  * 2、15位身份证无最后一位校验码。18位身份证中,校验码根据根据前17位生成
  * 
  * @param cardNumber
  * @return
  */
 private static String contertToNewCardNumber(String oldCardNumber) {
  StringBuilder buf = new StringBuilder(NEW_CARD_NUMBER_LENGTH);
  buf.append(oldCardNumber.substring(0, 6));
  buf.append("19");
  buf.append(oldCardNumber.substring(6));
  buf.append(CheckIdCard.calculateVerifyCode(buf));
  return buf.toString();
 }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值