当前访客身份:游客 [ 登录 | 加入 OSCHINA ]

代码分享

当前位置:
代码分享 » Java  » 常用工具方法
分享到: 
收藏 +0
3

最近项目中需要给用户增加身份证号字段,参考了几位别人的实现。

特点:1、面向对象:把身份证号封装为一个类,解析各个字段、验证有效性都是对象上的实例方法。对比那种公开多个静态方法的工具类的方式,我觉得这种面向对象的方式更自然一些。

2、不可变的。身份证号对象是不可变的,减少使用中的复杂性。

3、不是线程安全的。

标签: 精华

代码片段(1) [全屏查看所有代码]

1. [代码]公司、项目名称隐去了:)     跳至 [1] [全屏预览]

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

/**
 * 身份证号码,可以解析身份证号码的各个字段,以及验证身份证号码是否有效<br>
 * 身份证号码构成:6位地址编码+8位生日+3位顺序码+1位校验码
 * 
 * @author liuex
 * 
 */
public class IDCard {
	/**
	 * 完整的身份证号码
	 */
	private final String cardNumber;
	// 缓存身份证是否有效,因为验证有效性使用频繁且计算复杂
	private Boolean cacheValidateResult = null;
	// 缓存出生日期,因为出生日期使用频繁且计算复杂
	private Date cacheBirthDate = null;

	public boolean validate() {
		if (null == cacheValidateResult) {
			boolean result = true;
			// 身份证号不能为空
			result = result && (null != cardNumber);
			// 身份证号长度是18(新证)
			result = result && NEW_CARD_NUMBER_LENGTH == cardNumber.length();
			// 身份证号的前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;
			}
			// TODO 完整身份证号码的省市县区检验规则
			cacheValidateResult = Boolean.valueOf(result);
		}
		return cacheValidateResult;
	}

	/**
	 * 如果是15位身份证号码,则自动转换为18位
	 * 
	 * @param cardNumber
	 */
	public IDCard(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("身份证号码不正确!");
		}
	}

	// 身份证号码中的出生日期的格式
	private final static String BIRTH_DATE_FORMAT = "yyyyMMdd";
	// 身份证的最小出生日期,1900年1月1日
	private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L);
	private final static int NEW_CARD_NUMBER_LENGTH = 18;
	private final static int OLD_CARD_NUMBER_LENGTH = 15;
	/**
	 * 18位身份证中最后一位校验码
	 */
	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 };

	/**
	 * <li>校验码(第十八位数):<br/>
	 * <ul>
	 * <li>十七位数字本体码加权求和公式 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;</li>
	 * <li>计算模 Y = mod(S, 11)</li>
	 * <li>通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9 8 7 6 5 4 3 2</li>
	 * </ul>
	 * 
	 * @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(IDCard.calculateVerifyCode(buf));
		return buf.toString();
	}
}


开源中国-程序员在线工具:Git代码托管 API文档大全(120+) JS在线编辑演示 二维码 更多»

发表评论 回到顶部 网友评论(20)

  • 1楼:sunney888 发表于 2011-11-22 18:03 回复此评论
    不错。
  • 2楼:主编 发表于 2011-11-22 19:24 回复此评论

  • 3楼:梁援-晋 发表于 2011-11-23 09:30 回复此评论
     学习了  呵呵呵
  • 4楼:chenganshi 发表于 2011-11-25 19:53 回复此评论
    这个不错的,学习!
  • 5楼:xinyidt 发表于 2011-11-26 11:44 回复此评论
    这个不错,学习了!
  • 6楼:aonthly 发表于 2011-11-27 20:48 回复此评论
       嗯 。   不错  学习 了。
  • 7楼:kevinG 发表于 2011-11-28 05:35 回复此评论
    这个我有个PHP版本的!
  • 8楼:任民 发表于 2011-11-28 09:15 回复此评论
    对身份证的了解又有了进一步的加深。
  • 9楼:swingo 发表于 2011-12-29 23:09 回复此评论
    灰常不错!学习了!
  • 10楼:tonghuazhong 发表于 2012-02-01 09:44 回复此评论
    我就想知道其中的注释是怎么生产带<li>等这种标签的?
  • 11楼:dylan2080 发表于 2012-02-29 07:40 回复此评论
    灰常不错!学习了!
  • 12楼:夏悸 发表于 2012-08-03 13:00 回复此评论

    引用来自“tonghua zhong”的评论

    我就想知道其中的注释是怎么生产带<li>等这种标签的?
    自己手写的吧。。。
  • 13楼:叶国伟 发表于 2012-11-24 21:50 回复此评论
    看了一下。。这个方法validate用得有些不是太对。。太多重复了。。
  • 14楼:予沁安 发表于 2012-12-06 03:26 回复此评论
    在是用你的代码做重构的例子,有问题请告知 http://my.oschina.net/wonner/blog/94181
  • 15楼:liuex 发表于 2012-12-08 21:36 回复此评论

    引用来自“予沁安”的评论

    在是用你的代码做重构的例子,有问题请告知 http://my.oschina.net/wonner/blog/94181
    已经拜读并回复评论
  • 16楼:liuex 发表于 2012-12-08 21:41 回复此评论

    引用来自“叶国伟”的评论

    看了一下。。这个方法validate用得有些不是太对。。太多重复了。。
    validate实际上最多只会执行一次,然后就把结果缓存起来了。 多次调用checkIfValid()是为了尽早发现错误,发现错误后尽早终止。 如果身份证号码本身就是错误的,那么再去试图解析和提取身份证号中的信息就是完全没有意义的事情了。所以提取地址、性别等信息时,都会先检查身份证号码是否正确,如果不正确则立即终止。 提取生日的时候为何没有检验有效性,我自己也记不清为啥了……
  • 17楼:予沁安 发表于 2012-12-18 01:41 回复此评论

    引用来自“liuex”的评论

    引用来自“予沁安”的评论

    在是用你的代码做重构的例子,有问题请告知 http://my.oschina.net/wonner/blog/94181
    已经拜读并回复评论
    谢谢, 我非常同意你对validate的看法,只是实现方式和你的不一样。我是在构造器中验证。
  • 18楼:hellocpp 发表于 2012-12-19 11:38 回复此评论
    写得不错,学习了
  • 19楼:PaleC 发表于 2013-01-19 10:51 回复此评论
    有 台湾和香港 身份证 分析吗
  • 20楼:彭季 发表于 2013-12-03 09:23 回复此评论
    楼上有说在构造器中不好,还是楼主的方法不错
开源从代码分享开始 分享代码