MySql整理篇之身份证校验

规则描述

中国身份证分为一代15位和二代18位,排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。顺序码的奇数分给男性,偶数分给女性。校验码是根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校验码计算出来的检验码。详细规则参考百科身份证规则

场景

各种场景下会用到证件有效性校验,证件与性别、生日校验,现特整理编辑出MySql数据库环境下的证件校验,使用到自定义的两个函数:FUN_SPLIT(根据分隔符取字符串中的第几位)、FUN_VERIFY_CERT(身份证校验)。

详细设计

FUN_SPLIT函数

DELIMITER $$
DROP FUNCTION IF EXISTS FUN_SPLIT$$
CREATE FUNCTION FUN_SPLIT(P_INPUT VARCHAR(500),P_SPLIT VARCHAR(20),P_INDEX INT)
    RETURNS VARCHAR(50) CHARSET utf8
    BEGIN
    DECLARE RTV_VALUE VARCHAR(50) DEFAULT '';
    IF P_INPUT IS NULL THEN 
        RETURN NULL;
    END IF;
    IF (P_SPLIT IS NULL OR P_SPLIT = '') THEN 
        RETURN P_INPUT;
    END IF;
    SET RTV_VALUE := SUBSTRING_INDEX(SUBSTRING_INDEX(P_INPUT,P_SPLIT,P_INDEX),P_SPLIT,-1);
    RETURN RTV_VALUE;
    END$$
DELIMITER ;

FUN_VERIFY_CERT函数

DELIMITER $$
DROP FUNCTION IF EXISTS FUN_VERIFY_CERT$$
CREATE
    /*
    @Param ID_NUMBER 证件号
    @Param P_SEX 性别 0-女 1-男 可传入中文或码值
    @Param P_BIRTHDAY 生日
    备注:性别生日不校验可以传入NULL,生日格式为 yyyymmdd 或 yyyy-mm-dd
    */
    FUNCTION FUN_VERIFY_CERT(ID_NUMBER VARCHAR(20), P_SEX VARCHAR(1),P_BIRTHDAY VARCHAR(10))
    RETURNS VARCHAR(200) CHARSET utf8
    
    BEGIN
    DECLARE R_RETURN VARCHAR(50) DEFAULT '';-- 返回值
    DECLARE V_TEMP VARCHAR(50) DEFAULT '';-- 临时变量
    DECLARE V_HANDLER VARCHAR(50) DEFAULT 'OK';-- 临时变量
    DECLARE V_TMP_15_18 VARCHAR(50);-- 18 15位证件号
    DECLARE V_ID_SUM  BIGINT DEFAULT 0;-- 证件系数求和
    DECLARE V_ID_SEX VARCHAR(10) DEFAULT '';-- 证件性别
    DECLARE V_ID_BIRTH VARCHAR(10) DEFAULT '';-- 证件出生日期
    DECLARE V_ID_TEMP BIGINT; -- 证件号临时
    DECLARE V_IN_TEMP BIGINT; -- 系数临时
    DECLARE V_IM_TEMP BIGINT; -- mod 临时
    DECLARE V_INDEX BIGINT DEFAULT 1;-- 遍历索引
    DECLARE V_N VARCHAR(40) DEFAULT '7,9,10,5,8,4,2,1,6,3,7,9,10,5,8,4,2';-- 身份证系数
    DECLARE V_M VARCHAR(25) DEFAULT '1,0,X,9,8,7,6,5,4,3,2';-- 求MOD得的余数
    DECLARE CONTINUE HANDLER FOR 1292 SET V_HANDLER='日期格式不对';
    DECLARE CONTINUE HANDLER FOR 1690 SET V_HANDLER='校验字段过长';
    -- 判断非空
    IF (ID_NUMBER IS NULL OR ID_NUMBER = '') THEN 
        SET R_RETURN := '';
        RETURN R_RETURN;
    END IF;
    -- 判断长度
    IF (LENGTH(ID_NUMBER) <> 15 AND LENGTH(ID_NUMBER) <> 18) THEN
        SET R_RETURN := CONCAT(ID_NUMBER,'长度非[15,18]');
        RETURN R_RETURN;
    END IF;
    
    -- 15位证件号校验
    IF (LENGTH(ID_NUMBER) = 15) THEN
        BEGIN
            SET V_TMP_15_18 := CONCAT(SUBSTR(ID_NUMBER, 0, 6),'19',SUBSTR(ID_NUMBER, 7));
            -- 循环计算身份证前17位和权加因子的相乘得到的总合
            myloop:LOOP
                SET V_ID_TEMP := CAST(SUBSTR(V_TMP_15_18,V_INDEX,1) AS SIGNED);
                SET V_IN_TEMP := CAST(FUN_SPLIT(V_N,',',V_INDEX) AS SIGNED);
                
                SET V_ID_SUM := V_ID_SUM + V_ID_TEMP * V_IN_TEMP;
                
                SET V_INDEX := V_INDEX+1;
                
                IF V_INDEX > 17 THEN  LEAVE myloop;END IF;
            END LOOP myloop;
            -- 将得到的总合除以11得到一个余数,余数对应相应值
            SET V_IM_TEMP := FUN_SPLIT(V_M,',',MOD(V_ID_SUM,11) + 1);
            -- 校验位比较 15位不存在
            
            -- 性别取值:0-女 1-男
            SELECT CASE MOD((CAST(SUBSTR(ID_NUMBER, 15, 1) AS SIGNED)), 2) 
                   WHEN 0 THEN '0' ELSE '1' END INTO V_ID_SEX;
            -- 性别赋值
            IF V_ID_SEX = '0' THEN SET V_TEMP := '0-女'; END IF; 
            IF V_ID_SEX = '1' THEN SET V_TEMP := '1-男'; END IF;
            -- 传入性别默认:0-女 1-男 ,中文则转码转码
            IF (P_SEX IS NOT NULL AND P_SEX <> '') THEN 
                IF P_SEX = '女' THEN SET P_SEX:= '0'; END IF;
                IF P_SEX = '男' THEN SET P_SEX:= '1'; END IF;
                -- 比较性别
                IF (V_ID_SEX <> P_SEX) THEN
                    RETURN CONCAT(ID_NUMBER,'性别不匹配,证件号性别为:',V_TEMP);
                END IF;
            END IF;
            
            -- 出生时间取值
                SET @year  := SUBSTR(V_TMP_15_18, 1, 4);-- 年
                SET @month := SUBSTR(V_TMP_15_18, 5, 2);-- 月
                SET @day   := SUBSTR(V_TMP_15_18, 7, 2);-- 日
                SET V_ID_BIRTH := CONCAT(@year,'-',@month,'-',@day);
                SET V_TEMP := '';-- 恢复临时变量
                IF (P_BIRTHDAY IS NOT NULL AND P_BIRTHDAY <> '') THEN 
                BEGIN
                  SELECT DATE_FORMAT(P_BIRTHDAY,'%Y-%m-%d') INTO V_TEMP;
                  IF (V_TEMP IS NOT NULL AND V_TEMP <> '') THEN 
                    IF V_ID_BIRTH <> V_TEMP THEN
                        RETURN CONCAT(ID_NUMBER,'生日不匹配,证件生日为:',V_ID_BIRTH);
                    END IF;
                    IF V_ID_BIRTH = V_TEMP THEN 
                        -- return CONCAT('V_ID_BIRTH:',V_ID_BIRTH,',P_BIRTHDAY:',V_TEMP);
                        -- 身份其他信息校验
                        SET V_TEMP := '';
                    END IF;
                  ELSE 
                      RETURN CONCAT('生日参数:',P_BIRTHDAY,'格式(yyyy-mm-dd)不对');
                  END IF;
                END;
                END IF;
            -- 返回
            RETURN CONCAT(ID_NUMBER);
        END;
    END IF;
    
    -- 18位证件号校验
    IF (LENGTH(ID_NUMBER) = 18) THEN
        BEGIN   
            SET V_INDEX = 1;
            SET V_TEMP := '';           
            SET V_ID_SUM := 0;
            SET V_TMP_15_18 := ID_NUMBER;
            -- 循环计算身份证前17位和权加因子的相乘得到的总合
            myloop:LOOP
                SET V_ID_TEMP := CAST(SUBSTR(V_TMP_15_18,V_INDEX,1) AS SIGNED);
                SET V_IN_TEMP := CAST(FUN_SPLIT(V_N,',',V_INDEX) AS SIGNED);
                
                SET V_ID_SUM := V_ID_SUM + V_ID_TEMP * V_IN_TEMP;
                
                SET V_INDEX := V_INDEX+1;
                
                IF V_INDEX > 17 THEN  LEAVE myloop;END IF;
            END LOOP myloop;
            
            -- 将得到的总合除以11得到一个余数,余数对应相应值
            SET V_IM_TEMP := FUN_SPLIT(V_M,',',MOD(V_ID_SUM,11) + 1);
            SET V_TEMP := UPPER(SUBSTR(V_TMP_15_18, 18, 1));
            
            -- -- 校验位比较 第18位为校验位
            IF V_IM_TEMP <> V_TEMP THEN
                RETURN CONCAT(ID_NUMBER,'校验位不正确,',V_IM_TEMP,'!=',V_TEMP);
            END IF;
            
            -- 性别取值:0-女 1-男
            SELECT CASE MOD((CAST(SUBSTR(ID_NUMBER, 17, 1) AS SIGNED)), 2) 
                   WHEN 0 THEN '0' ELSE '1' END INTO V_ID_SEX;
            -- 性别赋值
            IF V_ID_SEX = '0' THEN SET V_TEMP := '0-女'; END IF; 
            IF V_ID_SEX = '1' THEN SET V_TEMP := '1-男'; END IF;
            -- 传入性别默认:0-女 1-男 ,中文则转码转码
            IF (P_SEX IS NOT NULL AND P_SEX <> '') THEN 
                IF P_SEX = '女' THEN SET P_SEX:= '0'; END IF;
                IF P_SEX = '男' THEN SET P_SEX:= '1'; END IF;
                -- 比较性别
                IF (V_ID_SEX <> P_SEX) THEN
                    RETURN CONCAT(ID_NUMBER,'性别不匹配,证件号性别为:',V_TEMP);
                END IF;
            END IF;
            
            -- 出生时间取值
                SET @year  := SUBSTR(V_TMP_15_18, 7, 4);-- 年
                SET @month := SUBSTR(V_TMP_15_18, 11,2);-- 月
                SET @day   := SUBSTR(V_TMP_15_18, 13,2);-- 日
                SET V_ID_BIRTH := CONCAT(@year,'-',@month,'-',@day);
                SET V_TEMP := '';-- 恢复临时变量
                IF (P_BIRTHDAY IS NOT NULL AND P_BIRTHDAY <> '') THEN 
                BEGIN
                  SELECT DATE_FORMAT(P_BIRTHDAY,'%Y-%m-%d') INTO V_TEMP;
                  IF (V_TEMP IS NOT NULL AND V_TEMP <> '') THEN 
                    IF V_ID_BIRTH <> V_TEMP THEN
                        RETURN CONCAT(ID_NUMBER,'生日不匹配,证件生日为:',V_ID_BIRTH);
                    END IF;
                    IF V_ID_BIRTH = V_TEMP THEN 
                        -- return CONCAT('V_ID_BIRTH:',V_ID_BIRTH,',P_BIRTHDAY:',V_TEMP);
                        -- 身份其他信息校验
                        SET V_TEMP := '';
                    END IF;
                  ELSE 
                      RETURN CONCAT('生日参数:',P_BIRTHDAY,'格式(yyyy-mm-dd)不对');
                  END IF;
                END;
                END IF;
            -- 返回
            RETURN CONCAT(ID_NUMBER);
        END;
    END IF;
    -- 最后一步无问题,返回本身证件号
    IF V_HANDLER <> 'OK' THEN 
        SET R_RETURN = V_HANDLER;
        RETURN R_RETURN;
    END IF;
    SET R_RETURN := ID_NUMBER;
    RETURN R_RETURN;
    END$$
    
DELIMITER ;

使用

以上两个函数直接Copy到数据库执行,在查询逻辑或者需要使用的地方直接使用函数FUN_VERIFY_CERT即可。例如:

SELECT FUN_VERIFY_CERT('142223198310300212',NULL,NULL) AS CheckResul;

备注:性别和生日若无需校验可传递NULL即可,生日参数格式为yyyymmdd或yyyy-mm-dd。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容