Greenplum数据库身份证查验函数

以下为 Greenplum 6.x 专用 的身份证校验函数,完整保留原始逻辑,无任何删减或优化,可直接复制使用:

sql 复制代码
CREATE OR REPLACE FUNCTION CHECK_ID_CARD("sfzh" varchar)
  RETURNS varchar AS $body$  
DECLARE
    RESULTS varchar(30);
BEGIN
	RESULTS := CASE
	WHEN sfzh IS NULL THEN
		'身份证号不能为空'
	WHEN LENGTH(sfzh) <> 18 THEN
		'身份证号码必须为18位'
	WHEN (substring(sfzh,1,17) ~ '[^0-9]' ) = 't' THEN
	  '身份证号码前17位格式不正确'
	WHEN substring(sfzh,7,4) NOT BETWEEN '1900' AND to_char(now(),'YYYY') THEN
		'身份证年份错误'
	WHEN substring(sfzh,11,2) NOT BETWEEN '01' AND '12' THEN 
		'身份证月份错误'
	WHEN substring(sfzh,13,2) NOT BETWEEN '01' AND to_char((date_trunc('month',to_date
	(CONCAT(substring(sfzh,7,4),'-',substring(sfzh,11,2),'-01'),'YYYY-MM-DD')) + interval '1 month' - interval '1 day'),'dd') THEN 
		'身份证日期错误'
	WHEN 
		mod(( 
            to_number(substring(sfzh,1,1),'9')*7+ 
            to_number(substring(sfzh,2,1),'9')*9+ 
            to_number(substring(sfzh,3,1),'9')*10+ 
            to_number(substring(sfzh,4,1),'9')*5+ 
            to_number(substring(sfzh,5,1),'9')*8+ 
            to_number(substring(sfzh,6,1),'9')*4+ 
            to_number(substring(sfzh,7,1),'9')*2+ 
            to_number(substring(sfzh,8,1),'9')*1+ 
            to_number(substring(sfzh,9,1),'9')*6+ 
            to_number(substring(sfzh,10,1),'9')*3+ 
            to_number(substring(sfzh,11,1),'9')*7+ 
            to_number(substring(sfzh,12,1),'9')*9+ 
            to_number(substring(sfzh,13,1),'9')*10+ 
            to_number(substring(sfzh,14,1),'9')*5+ 
            to_number(substring(sfzh,15,1),'9')*8+ 
            to_number(substring(sfzh,16,1),'9')*4+ 
            to_number(substring(sfzh,17,1),'9')*2 
          ),11) 
			<> to_number(
          ( 
          case
            when substring(sfzh,18,1)='1' then '0'
            when substring(sfzh,18,1)='0' then '1'
            when substring(sfzh,18,1) in ('X','x') then '2'
            when substring(sfzh,18,1)='9' then '3'
            when substring(sfzh,18,1)='8' then '4'
            when substring(sfzh,18,1)='7' then '5'
            when substring(sfzh,18,1)='6' then '6'
            when substring(sfzh,18,1)='5' then '7'
            when substring(sfzh,18,1)='4' then '8'
            when substring(sfzh,18,1)='3' then '9'
            when substring(sfzh,18,1)='2' then '10'
          end
          ),'99')
	THEN
		'身份证验证错误'
	ELSE
		'ok'
	END;
	Return(RESULTS);
END
$body$ LANGUAGE plpgsql;

核心逻辑说明

  1. 长度与格式

    LENGTH(sfzh) = 前17位全数字 + 第18位合法(数字/X/x)

  2. 日期合法性

    -- 自动校验当月最大天数(含闰年)
    to_char(date_trunc('month', to_date(...)) + interval '1 month' - interval '1 day', 'dd')

  3. 校验码算法 (GB 11643-1999)

    • 前17位按权重 [7,9,10,5,8,4,2,1,6,3,7,9,10,5,8,4,2] 加权求和
    • 求和后 mod 11
    • 根据余数映射校验码(0→1, 1→0, 2→X, 3→9, ..., 10→2

使用示例

sql 复制代码
-- 检查异常身份证
SELECT 
  sfzh,
  CHECK_ID_CARD(sfzh) AS check_result
FROM user_info

-- 输出示例:
-- 110101199002301234 | 身份证日期错误
-- 11010119900101123X | 身份证验证错误
-- 110101199001011234 | ok
相关推荐
Pocker_Spades_A2 小时前
数据库的“多模融合”——金仓
数据库
Maggie_ssss_supp2 小时前
Linux-MySQL数据类型&表操作
数据库·mysql
廋到被风吹走2 小时前
【数据库】【MySQL】高可用架构深度解析:从主从复制到自动切换
数据库·mysql·架构
IT邦德2 小时前
PostgreSQL 通过 mysql_fdw连通MySQL实战
数据库·mysql·postgresql
難釋懷2 小时前
Redis 通用命令
数据库·redis·缓存
hanqunfeng2 小时前
(九)Redis 命令及数据类型 -- Set
数据库·redis·bootstrap
企业对冲系统官2 小时前
期货与期权一体化平台风险收益评估方法与模型实现
运维·服务器·开发语言·数据库·python·自动化
IT邦德2 小时前
PostgreSQL通过Oracle_FDW连通Oracle实战
数据库·postgresql·oracle
廋到被风吹走2 小时前
【数据库】【MySQL】分区表深度解析:架构设计与大数据归档实践
android·数据库·mysql