转载自: 开源中国社区 面向对象的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("女性");
}
}
}
分享到:
相关推荐
为此,设计了一款专门用于校验15位与18位公民身份证号码合法性的SQL函数。 #### 二、身份证号码结构分析 根据中国公民身份证号码的标准,我们可以将身份证号码分为以下几部分: - **15位身份证号码**:由[6位地区码...
Oracle身份证号码校验函数,可以将15位身份号码升为18位,检验出生年月等信息
针对身份证号码的校验,以及能自动将身份证从15位升级为18位
根据给定的Linux shell脚本,我们可以深入探讨与身份证号码验证相关的知识点,特别是针对中国身份证号码的15位和18位格式的处理方法。以下是对脚本中关键部分的详细解析: ### Linux Shell编程验证身份证号码 ####...
pb验证身份证号输入的正确性(可以对身份证号码的第18位校验位进行判断) 源码 pb验证身份证号输入的正确性(可以对身份证号码的第18位校验位进行判断) 源码 pb验证身份证号输入的正确性(可以对身份证号码的第18位...
Java校验身份证号码工具类 ---运行几年了,很靠谱。包含15位身份证校验Java校验身份证号码工具类 ---运行几年了,很靠谱。包含15位身份证校验
可以校验15位和18位的身份证号码。 如果输入的18位身份证号码有错,将自动改为正确的号码。
身份证号码18位 和15位 验证的方法加正则表达式
身份证号码是中国公民重要的个人身份标识,它包含了个人的出生日期和地区的编码信息,并且通过一个校验码来确保号码的准确性。在Java编程中,验证身份证号码的校验码是否正确是一项常见的任务,特别是在处理个人信息...
本篇将详细探讨如何使用JavaScript进行15位和18位,特别是18位带有字母“X”的身份证号码的验证。 首先,我们需要理解身份证号码的结构。15位身份证号码由6位地区码、7位出生日期码和2位顺序码组成;18位身份证号码...
身份证号码由18位数字组成,按照以下规则排列: 1. **地址码**:前六位数字表示持证人常住户口所在地的县级行政区划代码,遵循GB/T 2260的规定。例如,“110101”代表北京市东城区。 2. **出生日期码**:接下来的...
中国的身份证号码(居民身份证)共18位,分为6个部分:地区码(前6位)、出生日期码(7到14位)、顺序码(15到17位,奇数分配给男性,偶数分配给女性)、校验码(最后一位)。校验码是通过特定算法计算得出的,用于...
简单校验主要检查身份证号码是否为15位或18位,并且对于18位号码,最后一位可以是数字或大写“X”。这种校验方式较为基础,适用于初步筛选或快速验证场景。以下是对应的正则表达式: ```javascript /^(\d{15}$|^\d{...
一个有效的18位身份证号码由6位地区代码、8位出生日期码和3位顺序码(奇数年出生的男性为偶数,偶数年出生的男性为奇数,女性反之)组成,最后一位是校验码,通过特定算法计算得出,确保了号码的唯一性和准确性。...
在这个函数中,首先检查输入的身份证号码长度是否为18位,然后根据权重数组计算和值,通过取模得到对应的校验字符。最后,比较输入身份证号码的第18位字符与计算出的校验字符是否一致,从而判断身份证号码的合法性。...
在中国大陆地区,身份证号码由18位数字组成,其中最后一位用于校验整个身份证号码的有效性。本文将详细介绍如何使用Java语言编写一个身份证号码校验类,并对其实现原理进行深入解析。 #### 二、身份证号码结构 ...
本函数提供18位身份证校验位的计算方法,输入前17位即可得到第18位的校验码!
在编程中实现身份证号码的校验,我们可以创建一个函数,输入18位的身份证号码,然后按照上述步骤进行校验。这个过程可以用于验证用户输入的身份证号码是否合法,或者在系统中存储和处理身份证号码时确保其准确性。 ...
标题中的“VB做的18位身份证号码真伪校验程序”指的是使用Visual Basic(VB)编程语言开发的一个软件工具,其主要功能是验证18位中国居民身份证号码的正确性和合法性。在中国,18位身份证号码是每个公民的唯一标识,...
甘肃人口全员管理系统(WIS系统)身份证号码批量校验.