最近项目中需要给用户增加身份证号字段,参考了几位别人的实现。
特点:
1、面向对象:把身份证号封装为一个类,解析各个字段、验证有效性都是对象上的实例方法。对比那种公开多个静态方法的工具类的方式,我觉得这种面向对象的方式更自然一些。
2、不可变的。身份证号对象是不可变的,减少使用中的复杂性。
3、不是线程安全的。

  1 import java.text.SimpleDateFormat;
  2 import java.util.Date;
  3
  4 /**
  5  * 身份证号码,可以解析身份证号码的各个字段,以及验证身份证号码是否有效<br>
  6  * 身份证号码构成:6位地址编码+8位生日+3位顺序码+1位校验码
  7  *
  8  * @author liuex
  9  *
 10  */
 11 public class IDCard {
 12     /**
 13      * 完整的身份证号码
 14      */
 15     private final String cardNumber;
 16     // 缓存身份证是否有效,因为验证有效性使用频繁且计算复杂
 17     private Boolean cacheValidateResult = null;
 18     // 缓存出生日期,因为出生日期使用频繁且计算复杂
 19     private Date cacheBirthDate = null;
 20
 21     public boolean validate() {
 22         if (null == cacheValidateResult) {
 23             boolean result = true;
 24             // 身份证号不能为空
 25             result = result && (null != cardNumber);
 26             // 身份证号长度是18(新证)
 27             result = result && NEW_CARD_NUMBER_LENGTH == cardNumber.length();
 28             // 身份证号的前17位必须是阿拉伯数字
 29             for (int i = 0; result && i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
 30                 char ch = cardNumber.charAt(i);
 31                 result = result && ch >= '0' && ch <= '9';
 32             }
 33             // 身份证号的第18位校验正确
 34             result = result
 35                     && (calculateVerifyCode(cardNumber) == cardNumber
 36                             .charAt(NEW_CARD_NUMBER_LENGTH - 1));
 37             // 出生日期不能晚于当前时间,并且不能早于1900年
 38             try {
 39                 Date birthDate = this.getBirthDate();
 40                 result = result && null != birthDate;
 41                 result = result && birthDate.before(new Date());
 42                 result = result && birthDate.after(MINIMAL_BIRTH_DATE);
 43                 /**
 44                  * 出生日期中的年、月、日必须正确,比如月份范围是[1,12],日期范围是[1,31],还需要校验闰年、大月、小月的情况时,
 45                  * 月份和日期相符合
 46                  */
 47                 String birthdayPart = this.getBirthDayPart();
 48                 String realBirthdayPart = this.createBirthDateParser().format(
 49                         birthDate);
 50                 result = result && (birthdayPart.equals(realBirthdayPart));
 51             } catch (Exception e) {
 52                 result = false;
 53             }
 54             // TODO 完整身份证号码的省市县区检验规则
 55             cacheValidateResult = Boolean.valueOf(result);
 56         }
 57         return cacheValidateResult;
 58     }
 59
 60     /**
 61      * 如果是15位身份证号码,则自动转换为18位
 62      *
 63      * @param cardNumber
 64      */
 65     public IDCard(String cardNumber) {
 66         if (null != cardNumber) {
 67             cardNumber = cardNumber.trim();
 68             if (OLD_CARD_NUMBER_LENGTH == cardNumber.length()) {
 69                 cardNumber = contertToNewCardNumber(cardNumber);
 70             }
 71         }
 72         this.cardNumber = cardNumber;
 73     }
 74
 75     public String getCardNumber() {
 76         return cardNumber;
 77     }
 78
 79     public String getAddressCode() {
 80         this.checkIfValid();
 81         return this.cardNumber.substring(0, 6);
 82     }
 83
 84     public Date getBirthDate() {
 85         if (null == this.cacheBirthDate) {
 86             try {
 87                 this.cacheBirthDate = this.createBirthDateParser().parse(
 88                         this.getBirthDayPart());
 89             } catch (Exception e) {
 90                 throw new RuntimeException("身份证的出生日期无效");
 91             }
 92         }
 93         return new Date(this.cacheBirthDate.getTime());
 94     }
 95
 96     public boolean isMale() {
 97         return 1 == this.getGenderCode();
 98     }
 99
100     public boolean isFemal() {
101         return false == this.isMale();
102     }
103
104     /**
105      * 获取身份证的第17位,奇数为男性,偶数为女性
106      *
107      * @return
108      */
109     private int getGenderCode() {
110         this.checkIfValid();
111         char genderCode = this.cardNumber.charAt(NEW_CARD_NUMBER_LENGTH - 2);
112         return (((int) (genderCode - '0')) & 0x1);
113     }
114
115     private String getBirthDayPart() {
116         return this.cardNumber.substring(6, 14);
117     }
118
119     private SimpleDateFormat createBirthDateParser() {
120         return new SimpleDateFormat(BIRTH_DATE_FORMAT);
121     }
122
123     private void checkIfValid() {
124         if (false == this.validate()) {
125             throw new RuntimeException("身份证号码不正确!");
126         }
127     }
128
129     // 身份证号码中的出生日期的格式
130     private final static String BIRTH_DATE_FORMAT = "yyyyMMdd";
131     // 身份证的最小出生日期,1900年1月1日
132     private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L);
133     private final static int NEW_CARD_NUMBER_LENGTH = 18;
134     private final static int OLD_CARD_NUMBER_LENGTH = 15;
135     /**
136      * 18位身份证中最后一位校验码
137      */
138     private final static char[] VERIFY_CODE = { '1', '0', 'X', '9', '8', '7',
139             '6', '5', '4', '3', '2' };
140     /**
141      * 18位身份证中,各个数字的生成校验码时的权值
142      */
143     private final static int[] VERIFY_CODE_WEIGHT = { 7, 9, 10, 5, 8, 4, 2, 1,
144             6, 3, 7, 9, 10, 5, 8, 4, 2 };
145
146     /**
147      * <li>校验码(第十八位数):<br/>
148      * <ul>
149      * <li>十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0...16 ,先对前17位数字的权求和;
150      * Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
151      * 2;</li>
152      * <li>计算模 Y = mod(S, 11)</li>
153      * <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>
154      * </ul>
155      *
156      * @param cardNumber
157      * @return
158      */
159     private static char calculateVerifyCode(CharSequence cardNumber) {
160         int sum = 0;
161         for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
162             char ch = cardNumber.charAt(i);
163             sum += ((int) (ch - '0')) * VERIFY_CODE_WEIGHT[i];
164         }
165         return VERIFY_CODE[sum % 11];
166     }
167
168     /**
169      * 把15位身份证号码转换到18位身份证号码<br>
170      * 15位身份证号码与18位身份证号码的区别为:<br>
171      * 1、15位身份证号码中,"出生年份"字段是2位,转换时需要补入"19",表示20世纪<br>
172      * 2、15位身份证无最后一位校验码。18位身份证中,校验码根据根据前17位生成
173      *
174      * @param cardNumber
175      * @return
176      */
177     private static String contertToNewCardNumber(String oldCardNumber) {
178         StringBuilder buf = new StringBuilder(NEW_CARD_NUMBER_LENGTH);
179         buf.append(oldCardNumber.substring(0, 6));
180         buf.append("19");
181         buf.append(oldCardNumber.substring(6));
182         buf.append(IDCard.calculateVerifyCode(buf));
183         return buf.toString();
184     }
185 }

转载于:https://www.cnblogs.com/Devin-Blog/p/5492077.html

面向对象的15、18位中国大陆身份证号码解析、工具相关推荐

  1. 代码整洁之道------它山之玉可以重构:身份证号码解析、验证工具(第一天)

    首先感谢原作者的代码: < 面向对象的15位.18位中国大陆身份证号码解析.验证工具>. 如果有版权问题,敬请告知. 原代码给了我很好的起点,是的,这些代码都是玉,我在这只是再琢磨琢磨. ...

  2. 15/18位身份证号码验证的正则表达式总结(详细版)

    前言 开发过程中,常常需要对一些输入信息进行有效性验证,使用正则表达式进行校验是最简单.效率最高的方式了,下面就来看看15/18位身份证号码验证的正则表达式吧. 介绍 xxxxxx yyyy MM d ...

  3. 15/18位身份证号码验证

    15/18位身份证号码验证 在做身份证验证之前你得明确身份证号的组成. 首先我告诉你,18位身份证号码后面的字母只有"X"这一个.(X是罗马数字的10,用X来代替10,可以保证您的 ...

  4. 中国公民身份证号码校验

    中国公民身份证号码校验 上图可看出18位位身份证每一部分表示的意义. 图为随机生成身份号码,四川省 乐山市 井研县 1981年12月28日生 这里主要讲解最后一位校验码的验证 校验码主要是由前面17位 ...

  5. Js实现中国公民身份证号码有效性验证

    2019独角兽企业重金招聘Python工程师标准>>> 参考: Java实现中国公民身份证号码有效性验证 Php实现中国公民身份证号码有效性验证 本文将使用JavaScript实现中 ...

  6. Java 身份证号码验证工具类

    package cn.hlq.test;import java.text.ParseException; import java.text.SimpleDateFormat; import java. ...

  7. python读取身份证号_Python实现身份证号码解析

    中国的居民身份证有18位.其中前17位是信息码,最后1位是校验码.每位信息码可以是0-9的数字,而校验码可以是0-9或X,其中X表示10. 身份证校验码算法: 设18位身份证号序列从左到右为: 引用 ...

  8. python身份证号码解析编程_[宜配屋]听图阁

    中国的居民身份证有18位.其中前17位是信息码,最后1位是校验码.每位信息码可以是0-9的数字,而校验码可以是0-9或X,其中X表示10. 身份证校验码算法: 设18位身份证号序列从左到右为: 引用 ...

  9. 中国身份证号码验证,支持15,18位,可验证成功90%的身份证号

    using System;/// <summary> /// Summary description for IDCardValid /// </summary> public ...

  10. Jquery 对 身份证号码的验证 (15/18位)

    //欢迎验证.给予指正错误,从好多地方查来的资料修改的...已用本人身份证验证... 由于以前修改的问题无法验证带 X 的身份证号码,已修复!//为值添加0function Append_zore(t ...

最新文章

  1. Linux下修改Mysql的用户(root)的密码
  2. pyCharm编辑器激活使用
  3. python常用内置函数总结-Python学习教程之常用的内置函数大全
  4. POJ1269 直线相交
  5. Theano mnist数据集格式
  6. 服务器的系统如何入账,云服务器费用如何入账
  7. Spark SQL(三)之视图与执行SQL
  8. 今天maven install时碰到的两个问题(堆溢出和编译错误)
  9. html文件怎么保存链接,如何使用beautifulsoup将链接的html保存在文件中,并对html文件中的所有链接执行相同的操作...
  10. JQuery插件iScroll实现下拉刷新,滚动翻页特效
  11. CC3200学习——blinky资料
  12. GoGoCode 代码语言转换
  13. tensorflow graphics详解
  14. prop()方法的应用
  15. MySQL日志管理、备份与恢复
  16. ROS1/2 机器人编程实践汇总 kinetic/melodic/noetic foxy/galactic/humble
  17. 对谈 | 创新与进化——当开源接受SaaS
  18. android studio提示HAXM device is not found或者could not start avd
  19. 【距离高考还有多少天】
  20. Linux进程间通信IPC

热门文章

  1. web编辑器——百度UEditor编辑器使用教程与使用方法
  2. 施一公等团队登Science封面:AI与冷冻电镜揭示「原子级」NPC结构,生命科学突破...
  3. 微信公众号跳转微信小程序,自定义微信跳转标签
  4. mysql rownum写法_mysql类似oracle rownum写法
  5. 计算机学院研发------考核之界面
  6. 电力行业数据安全解决方案
  7. 直播预告 | AAAI 2022:一种基于能量的主动域自适应学习方法
  8. C++回调函数使用心得
  9. e5服务器cpu性能排行,了解至强cpu!至强cpu性能排行榜及具体参数详解【详解】...
  10. python 3des加密_Python 3DES 加密解密