JAVA验证身份证号码是否合法

package com.chauvet.utils;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/***
 * 
 * 身份证18位分别代表的含义,身份证15位升级到18位,原来年用2位且没有最后一位,从左到右方分别表示
 * ①1-2 升级行政区代码
 * ②3-4 地级行政区划分代码
 * ③5-6 县区行政区分代码
 * ④7-10 11-12 13-14 出生年、月、日
 * ⑤15-17 顺序码,同一地区同年、同月、同日出生人的编号,奇数是男性,偶数是女性
 * ⑥18 校验码,如果是0-9则用0-9表示,如果是10则用X(罗马数字10)表示(最后一位可能出现的X并不是英文字母X,而是希腊数字10的缩写X)
 * 
 * 只要将每位的对应权重乘以每个位上的数值,然后求和,最后与11求余数,得到的结果对比找到尾数即可。
 * 
 * 原有15位身份证是没有校验位的,同时采用的是2位数字来表示出生年份
 * 
 * 
 * 
 * 第一步:
 * 先对前17位数字的权求和 ,使用十七位数字本体码加权求和公式   S = Sum(Ai * Wi), i = 0, ... , 16
 * Ai:表示第i位置上的身份证号码数字值(0~9) 
 * Wi:7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2 (固定的,表示第i位置上的加权因子)
 * 将身份证号码的第1位数字与7相乘;
 * 将身份证号码的第2位数字与9相乘;
 * 将身份证号码的第3位数字与10相乘;
 * 将身份证号码的第4位数字与5相乘;
 * 将身份证号码的第5位数字与8相乘;
 * 将身份证号码的第6位数字与4相乘;
 * 将身份证号码的第7位数字与2相乘;
 * 将身份证号码的第8位数字与1相乘;
 * 将身份证号码的第9位数字与6相乘;
 * 将身份证号码的第10位数字与3相乘;
 * 将身份证号码的第11位数字与7相乘;
 * 将身份证号码的第12位数字与9相乘;
 * 将身份证号码的第13位数字与10相乘;
 * 将身份证号码的第14位数字与5相乘;
 * 将身份证号码的第15位数字与8相乘;
 * 将身份证号码的第16位数字与4相乘;
 * 将身份证号码的第17位数字与2相乘。
 * 
 * 第二步:
 * 将第一步身份证号码1~17位相乘的结果求和,全部加起来。
 * 
 * 第三步:
 * 计算模  Y = mod(S, 11)
 * 用第二步计算出来的结果除以11,这样就会出现
 * 余数为0,
 * 余数为1,
 * 余数为2,
 * 余数为3,
 * 余数为4,
 * 余数为5,
 * 余数为6,
 * 余数为7,
 * 余数为8,
 * 余数为9,
 * 余数为10
 * 共11种可能性。
 * 
 * 第四步:
 * 根据模,查找得到对应的校验码 
 * Y:    0 1 2 3 4 5 6 7 8 9 10 
 * 校验码: 1 0 X 9 8 7 6 5 4 3 2
 * 如果余数为0,那对应的最后一位身份证的号码为1;
 * 如果余数为1,那对应的最后一位身份证的号码为0;
 * 如果余数为2,那对应的最后一位身份证的号码为X;
 * 如果余数为3,那对应的最后一位身份证的号码为9;
 * 如果余数为4,那对应的最后一位身份证的号码为8;
 * 如果余数为5,那对应的最后一位身份证的号码为7;
 * 如果余数为6,那对应的最后一位身份证的号码为6;
 * 如果余数为7,那对应的最后一位身份证的号码为5;
 * 如果余数为8,那对应的最后一位身份证的号码为4;
 * 如果余数为9,那对应的最后一位身份证的号码为3;
 * 如果余数为10,那对应的最后一位身份证的号码为2。
 * 
 * @author WXW
 *
 */
public class IDValidate {
    
    static String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7","9", "10", "5", "8", "4", "2" };//十七位数字本体码权重
    
    static String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4","3", "2" }; //最后一位的校验码字符值    
    
    public static void main(String[] args) {
        
        check("41272519910506****");
        
    }
    
    /***
     * 验证身份证
     * @param idStr
     * @return
     */
    public static boolean check(String idStr){
        try {
            if(null == idStr){// 验证非空
                System.out.println("身份证号码不能为空");
                return false;
            }
            if(idStr.length() != 15 && idStr.length() != 18){// 只能是15位或者18位
                System.out.println("身份证号码长度只能是15位或者18位");
                return false;
            }
            
             String Ai = "";
             if (idStr.length() == 18) {
                Ai = idStr.substring(0, 17);
            } 
             
             if (idStr.length() == 15) {//  将15位身份证转换为 17位身份证,最后加上最后一位,转换为18位身份证
                // 15位身份证是没有校验位的,同时采用的是2位数字来表示出生年份,并且 15位的身份证号码确定都是19**年的
                Ai = idStr.substring(0, 6) + "19" + idStr.substring(6, 15); 
            }
            if(!isNumber(Ai)){ // 验证身份证前17位是否都是数字
                System.out.println("身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字。");
                return false;
            }
            
            int year = Integer.parseInt(Ai.substring(6, 10));// 出生年份
            int month = Integer.parseInt(Ai.substring(10, 12));// 出生月份
            int day = Integer.parseInt(Ai.substring(12, 14));// 出生日期
            String birthDay = year+"-"+month+"-"+day;
            
            Date birthdate = null;
            try {// 将出生日期转换为Date类型
                birthdate = new SimpleDateFormat("yyyyMMdd").parse(birthDay);
            } catch (ParseException e) {
                e.printStackTrace();
                System.out.println("身份证生日无效。");
                return false;
            }
            if (birthdate == null || new Date().before(birthdate)) {
                System.out.println("身份证生日无效。");
                return false;
            }
            
            GregorianCalendar gc = new GregorianCalendar();//GregorianCalendar 是 Calendar 的一个具体子类,提供了世界上大多数国家/地区使用的标准日历系统。
            SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
            // 验证生日年份是否在范围之内
            if ((gc.get(Calendar.YEAR) - year) > 150 || (gc.getTime().getTime() - s.parse(birthDay).getTime()) < 0) {
                System.out.println("身份证生日不在有效范围。");
                return false;
            }
            
            //验证月份
            if (month > 12 || month <= 0) {
                System.out.println("身份证号中的月份无效");
                return false;
            }
            
            //验证日期
            gc.setTime(birthdate);
            boolean mflag = false;
            switch (month) {
            case 1:
            case 3:
            case 5:
            case 7:
            case 8:
            case 10:
            case 12:
                mflag = (day >= 1 && day <= 31);
                break;
            case 2: // 公历的2月非闰年有28天,闰年的2月是29天。
                if (gc.isLeapYear(gc.get(Calendar.YEAR))) {
                    mflag = (day >= 1 && day <= 29);
                } else {
                    mflag = (day >= 1 && day <= 28);
                }
                break;
            case 4:
            case 6:
            case 9:
            case 11:
                mflag = (day >= 1 && day <= 30);
                break;
            }
            if (!mflag) {// 日期不对
                System.out.println("省份证号中的出生日期不对");
                return false;
            }
            // 验证 开头两位数是否是真实有效的地区编码
            if (cityCodeMap.get(Ai.substring(0, 2)) == null) {
                System.out.println("身份证地区编码错误。");
                return false;
            }
            
            int TotalmulAiWi = 0;
            for (int i = 0; i < 17; i++) {//先对前17位数字的权求和 ,使用十七位数字本体码加权求和公式   S = Sum(Ai * Wi)
                TotalmulAiWi = TotalmulAiWi + Integer.parseInt(String.valueOf(Ai.charAt(i))) * Integer.parseInt(Wi[i]);
            }
            int modValue = TotalmulAiWi % 11;//计算模  Y = mod(S, 11) 用计算出来的结果除以11,这样就会出现
            String strVerifyCode = ValCodeArr[modValue]; // 获取最后一位的校验码字符值    
            Ai = Ai + strVerifyCode; // 17位身份证 加上最后以为验证数字   得到18位有效的身份证号
            
            if(!idStr.toUpperCase().equals(Ai.toUpperCase())){// 判断传过来的身份证号 和 计算得到的身份证号是否相同
                System.out.println("身份证号码不对");
                return false;
            }
            System.out.println("正确");
            return true;
        } catch (Exception e) {
            System.out.println("验证出错");
            e.printStackTrace();
            return false;
        }
        
        
    }
    
    /***
     * 地区编码
     */
    private static Map<String, String> cityCodeMap = new HashMap<String, String>() {
        {
            this.put("11", "北京");
            this.put("12", "天津");
            this.put("13", "河北");
            this.put("14", "山西");
            this.put("15", "内蒙古");
            this.put("21", "辽宁");
            this.put("22", "吉林");
            this.put("23", "黑龙江");
            this.put("31", "上海");
            this.put("32", "江苏");
            this.put("33", "浙江");
            this.put("34", "安徽");
            this.put("35", "福建");
            this.put("36", "江西");
            this.put("37", "山东");
            this.put("41", "河南");
            this.put("42", "湖北");
            this.put("43", "湖南");
            this.put("44", "广东");
            this.put("45", "广西");
            this.put("46", "海南");
            this.put("50", "重庆");
            this.put("51", "四川");
            this.put("52", "贵州");
            this.put("53", "云南");
            this.put("54", "西藏");
            this.put("61", "陕西");
            this.put("62", "甘肃");
            this.put("63", "青海");
            this.put("64", "宁夏");
            this.put("65", "新疆");
            this.put("71", "台湾");
            this.put("81", "香港");
            this.put("82", "澳门");
            this.put("91", "国外");
        }
    };

    /***
     * 判断str是否为纯数字组成
     * @param str
     * @return
     */
    private static boolean isNumber(String str) {
        Pattern pattern = Pattern.compile("[0-9]*");
        Matcher isNum = pattern.matcher(str);
        return isNum.matches();
    }
}

 

posted @ 2016-08-09 17:18  wxw_wang  阅读(987)  评论(0)    收藏  举报
/* 下雪 begin */ /* 下雪 end */ /* 点击出现爱心特效 begin*/ /* 点击出现爱心特效 end*/