【这是一猿小讲的第 28 篇原创分享】
1.
鲁迅先生说:程序员,天不怕地不怕,就怕小虫儿爬呀爬,爬呀爬。
随着时间的推移,鲁迅先生又说: 真正勇猛的程序员,敢于让虫子面对惨淡的虫生。
虫子在 程序员心中是啥东西 ? 虫子的学名为 Bug,是多少入了猿门的程序员,心中不愿提及的痛。 消灭虫子的技能为 Debug,这项技能我敢保证: 无论你是小白、大牛还是骨灰级的程序员,多多少少都需要进行修炼。
千年虫又是啥东东? 计算机 2000 年问题,又叫做“千年虫”、“电脑千禧年千年虫问题”或“千年危机”。 缩写为“Y2K”。 是指在某些使用了计算机程序的智能系统(包括计算机系统、自动控制芯片等)中,由于其中的年份只使用两位十进制数来表示,因此当系统进行(或涉及到)跨世纪的日期处理运算时(如多个日期之间的计算或比较等),就会出现错误的结果,进而引发各种各样的系统功能紊乱甚至崩溃。 因此从根本上说千年虫是一种程序处理日期上的 bug(计算机程序故障) —— 摘自百度百科 。
看完官方的回答,那我们能否用人话再解释一下“ 千年虫 ”? 其实大概意思是这样的: 由于之前计算机内存比较小 ,时间只用 2 位数来进行表示“年”,例如 1998 年就表示成 98。 但是到了 2000 年,机器上出现的时间就是 00 年,而电脑就不能识别,这个是公元 1900 年,还是公元 2000 年。就好比 你在 1999 年 12 月 31 日 23: 59 分打了三分钟的电话,电话局的账单却可能显示为 (-100年+3分钟), 这就是所谓的“ 千年虫的问题 ”。
好了,准备好小板凳,今天的故事就从虫子(Bug)说起。
2.
前些天,Python 菇凉跟着 Java 那小子,学会了安装、破解 IDEA,并熟练用 IDEA 编写 HelloWorld 入门程序。 为了庆祝 Python 菇凉的顺利步入猿门,Java 那小子带着 Python 菇凉来到了著名而又浪漫的 C++C-- 游乐场。
C++C-- 游乐场确实名气很大,购买入场票的队伍,像闹市里的车水马龙一样挤得满满的。
时间过去了两个小时,终于轮到了 Java 那小子 & Python 菇凉。 Java 那小子把身份证递给售票员 Node.js 小姐姐。
售票员 Node.js 小姐姐说: Python 菇凉你的身份证号为 15 位,购票系统提示“身份证号位数不足 18 位”,麻烦请提供 18 位的身份证号。
Java 那小子心中有点浴火,不过这也不能怪 Node.js 小姐姐,只能说他们的系统不支持自动把 15 位身份证号自行变为 18 位身份证号,场面很尴尬,那需要赶紧想个招来。
在猿门摸爬滚打久了的 Java 那小子,无论去哪儿都会随身携带电脑,以防不测。 说时迟那时快,只见 Java 那小子已经打开笔记本电脑,猛如虎的请教度娘 & 谷哥,关于身份证规则的前世今生。
早期身份证由 15 位数字构成,这主要是在 1980 年以前发放的身份证,其中编排规则如下。
前 1、2 位数字表示:所在省份的代码;
第 3、4 位数字表示:所在城市的代码;
第 5、6 位数字表示:所在区县的代码;
第 7~12 位数字表示:两位年、月、日(yyMMdd);
第 15 位数字表示性别:奇数表示男性,偶数表示女性。
聪明的你肯定也会发现 15 位的身份证号,只能为 1900 年 1 月 1 日到 1999 年 12 月 31 日出生的人编号,如果是 2019 年出生的,按照 15 位身份证规则,应该记为 19 ,但是这个到底代表 1919 还是 2019呢? 这明显存在一个千年虫子的问题,考虑到这个 Bug,因而又增加了 18 位身份证号码编号规则。
前 1、2 位数字表示:所在省份的代码;
第 3、4 位数字表示:所在城市的代码;
第 5、6 位数字表示:所在区县的代码;
第 7~14 位数字表示:出生年、月、日(yyyyMMdd);
第 15~17 位数字是顺序码:表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号,顺序码的奇数分配给男性,偶数分配给女性(即第17位是奇数代表男性,是偶数代表女性)。
第 18 位数字是校检码:用来检验身份证的正确性。校检码可以是 0~9 的数字,有时也用 X 表示。
15位 vs 18 位身份证号区别在哪里?
15 位身份证号出生年月日为 yyMMdd,而 18 位身份证号格式为yyyyMMdd;
15 位身份证号没有校验码。
我们该如何实现 15位 转成 18 位的身份证号呢?
0、首先将15位的身份证号,加入两位年份变成17位身份证号;
1、将前面的身份证号码17位数分别乘以不同的系数。从第一位到第十七位的系数分别为:7-9-10-5-8-4-2-1-6-3-7-9-10-5-8-4-2。
2、将这17位数字和系数相乘的结果相加。
3、用加出来和除以11,看余数是多少?
4、余数只可能有0-1-2-3-4-5-6-7-8-9-10这11个数字。其分别对应的最后一位身份证的号码为1-0-X -9-8-7-6-5-4-3-2。(即余数0对应1,余数1对应0,余数2对应X...)
4.1、如果余数是3,就会在身份证的第18位数字上出现的是9。如果对应的数字是2,身份证的最后一位号码就是罗马数字X。
搞定了身份证编排规则后,再来码代码消灭这个千年虫就变得容易了。 只见 Java 那小子开启了身份证 15 位转换 18 位的行云流水般的编码操作。
经过 Java 那小子近 2 分钟的猛如虎的实现下,成功实现 Python 菇凉身份证 15 位转换成 18 位,然后把 18 位的证件号码给 Node.js 售票员小姐姐提供了一下,意料之中一次性购票成功。
Java 那小子处于他人考虑,把代码包装成一键运行的工具,分享给 Node.js 售票员小姐姐,希望能解决他人的燃眉之急。
至此,身份证千年虫的问题,被Java 那小子彻底消灭。Java 那小子与心爱的 Python 菇凉正常愉快的进入了 C++C-- 游乐场,开启了美好一天的浪漫玩耍。
3.
故事讲完了,由于身份证编排规则的更新,很多老的系统依然会存在 15 位的身份证号,此时要么从库中,批量更新用户的身份证号;要么进行扩展程序支持。其中无论你是用数据库批量更新;还是用程序跑批,15 位转成 18 位的身份证号的实现思路都是一样的。
好了,今天的分享就到这儿吧,希望对你能有帮助。
推荐阅读:
爱情36技之追美妹的技术
爱情36技之暗送秋波的技术