Java校验身份证号码合法性

来源:互联网 发布:access数据库64位下载 编辑:程序博客网 时间:2024/06/04 18:38
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.Hashtable;
import java.util.regex.Matcher;
import java.util.regex.Pattern;


/** 
 * 作用:校验身份证是否合法
 * 身份证号码的格式:610821-20061222-612-X 
 * 由18位数字组成:前6位为地址码,第7至14位为出生日期码,第15至17位为顺序码, 
 * 第18位为校验码。检验码分别是0-10共11个数字,当检验码为“10”时,为了保证公民身份证号码18位,所以用“X”表示。虽然校验码为“X”不能更换,但若需全用数字表示,只需将18位公民身份号码转换成15位居民身份证号码,去掉第7至8位和最后1位3个数码。  
 * 当今的身份证号码有15位和18位之分。1985年我国实行居民身份证制度,当时签发的身份证号码是15位的,1999年签发的身份证由于年份的扩展(由两位变为四位)和末尾加了效验码,就成了18位。 
 * 依照《中华人民共和国居民身份证法》规定,依照《中华人民共和国居民身份证条例》领取的居民身份证,自2013年1月1日起停止使用,在2012年1月1日以前领取的居民身份证,在其有效期内,继续有效,因此还可能会有部分有效身份证为15位数。
 * 各数位数字含义:
 * (1)前1、2位数字表示:所在省份的代码;  
 * (2)第3、4位数字表示:所在城市的代码; 
 * (3)第5、6位数字表示:所在区县的代码; 
 * (4)第7~14位数字表示:出生年、月、日; 
 * (5)第15、16位数字表示:所在地的派出所的代码;  
 * (6)第17位数字表示性别:奇数表示男性,偶数表示女性 
 * (7)第18位数字是校检码:根据一定算法生成 
 * @author jiangfei
 * 
 */  
public class CheckIdcardIsLegal {


public static String checkIdCard(String Idcard){

String mesgStr = "";
String Id = "";
//最后位为X或x时统一大写
Idcard = Idcard.toUpperCase();
//判断号码的长度为18位
if(Idcard.length()!=15 && Idcard.length()!=18){
mesgStr = "身份证号码位数应该为15位或18位!";
return mesgStr;
}
//如果为15位,则所有号码都为数字,如果为18位,前17位为数字
if(Idcard.length()==15){
//将15位身份证转换为17位数字
Id = Idcard.substring(0, 6) + "19" + Idcard.substring(6, 15);  

}else if(Idcard.length()==18){
Id = Idcard.substring(0, 17);
}

if(!isNumeric(Id)){
mesgStr = "身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字!";
return mesgStr;
}

// 判断出生年月是否有效   
        String strYear = Id.substring(6, 10);// 年份  
        String strMonth = Id.substring(10, 12);// 月份  
        String strDay = Id.substring(12, 14);// 日期  
        if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {  
        mesgStr = "身份证出生日期无效!";  
            return mesgStr;  
        }  
        GregorianCalendar gc = new GregorianCalendar();  
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");  
        try {  
            if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150  
                    || (gc.getTime().getTime() - sdf.parse(  
                            strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {  
            mesgStr = "身份证生日不在有效范围!";  
                return mesgStr;  
            }  
        } catch (NumberFormatException e) {  
            e.printStackTrace();  
        } catch (java.text.ParseException e) {  
            e.printStackTrace();  
        }  
        if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {  
        mesgStr = "身份证月份无效!";  
            return mesgStr;  
        }  
        if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {  
        mesgStr = "身份证日期无效!";  
            return mesgStr;  
        }  

        // 判断地区码是否有效   
        Hashtable<String,String> areacode = GetAreaCode();  
        //如果身份证前两位的地区码不在Hashtable,则地区码有误  
        if (areacode.get(Id.substring(0, 2)) == null) {  
            mesgStr = "身份证地区编码错误!";  
            return mesgStr;  
        }  
          
        if(isVarifyCode(Id,Idcard)==false){  
        mesgStr = "身份证校验码无效,不是合法的身份证号码!";  
            return mesgStr;  
        }  
return "yes";
}

/**判断第18位校验码是否正确 
* 第18位校验码的计算方式:  
* 1. 对前17位数字本体码加权求和  
* 公式为:S = Sum(Id[i]* WT[i]), i = 0, ... , 16  
* 其中Id表示第i个位置上的身份证号码数字值,Wi表示第i位置上的加权因子,其各位对应的值依次为: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2  
* 2. 用11对计算结果取模  
* Y = mod(S, 11)  
* 3. 根据模的值得到对应的校验码  
* 对应关系为:  
* Y值:     0  1  2  3  4  5  6  7  8  9  10  
* 校验码:  1  0  X  9  8  7  6  5  4  3   2 
    */  
   private static boolean isVarifyCode(String Id,String IDStr) {  
        String[] VarifyCode = { "1", "0", "X", "9", "8", "7", "6", "5", "4","3", "2" };  
        String[] WT = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7","9", "10", "5", "8", "4", "2" };  
        int sum = 0;  
        for (int i = 0; i < 17; i++) {  
           sum = sum + Integer.parseInt(String.valueOf(Id.charAt(i))) * Integer.parseInt(WT[i]);  
        }  
        int modValue = sum % 11;  
        String strVerifyCode = VarifyCode[modValue];  
       Id = Id + strVerifyCode;  
        if (IDStr.length() == 18) {  
            if (Id.equals(IDStr) == false) {  
                return false;  
            }  
        }   
        return true;  
   }  
/** 
     * 判断字符串是否为数字,0-9重复0次或者多次    
     * @param  
     * @return 
     */  
    private static boolean isNumeric(String strnum) {  
        Pattern pattern = Pattern.compile("[0-9]*");  
        Matcher isNum = pattern.matcher(strnum);  
        if (isNum.matches()) {  
            return true;  
        } else {  
            return false;  
        }  
    } 

    /** 
     * 判断字符串出生日期是否符合正则表达式:包括年月日,闰年、平年和每月31天、30天和闰月的28天或者29天 
     * @param string 
     * @return 
     */  
    public static boolean isDate(String strDate) {  
      
        Pattern pattern = Pattern.compile("^((\\d{2}(([02468][048])|([13579][26]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|" +
                "([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|" +
                "([1-2][0-9])))))|(\\d{2}(([02468][1235679])|([13579][01345789]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|" +
                "([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|(1[0-9])|(2[0-8]))))))?$");  
        Matcher m = pattern.matcher(strDate);  
        if (m.matches()) {  
            return true;  
        } else {  
            return false;  
        }  
    }  

    /** 
     * 将所有地址编码保存在一个Hashtable中     
     * @return Hashtable对象 
     */  
    private static Hashtable<String,String> GetAreaCode() {  
        Hashtable<String,String> hashtable = new Hashtable<String,String>();  
        hashtable.put("11", "北京");  
        hashtable.put("12", "天津");  
        hashtable.put("13", "河北");  
        hashtable.put("14", "山西");  
        hashtable.put("15", "内蒙古");  
        hashtable.put("21", "辽宁");  
        hashtable.put("22", "吉林");  
        hashtable.put("23", "黑龙江");  
        hashtable.put("31", "上海");  
        hashtable.put("32", "江苏");  
        hashtable.put("33", "浙江");  
        hashtable.put("34", "安徽");  
        hashtable.put("35", "福建");  
        hashtable.put("36", "江西");  
        hashtable.put("37", "山东");  
        hashtable.put("41", "河南");  
        hashtable.put("42", "湖北");  
        hashtable.put("43", "湖南");  
        hashtable.put("44", "广东");  
        hashtable.put("45", "广西");  
        hashtable.put("46", "海南");  
        hashtable.put("50", "重庆");  
        hashtable.put("51", "四川");  
        hashtable.put("52", "贵州");  
        hashtable.put("53", "云南");  
        hashtable.put("54", "西藏");  
        hashtable.put("61", "陕西");  
        hashtable.put("62", "甘肃");  
        hashtable.put("63", "青海");  
        hashtable.put("64", "宁夏");  
        hashtable.put("65", "新疆");  
        hashtable.put("71", "台湾");  
        hashtable.put("81", "香港"); 
        hashtable.put("82", "澳门");  
        hashtable.put("91", "国外");  
        return hashtable;  
    }
    
    public static void main(String[] args) {
    String string = CheckIdcardIsLegal.checkIdCard("610821200612225123");
    System.out.println(string);
}
    
}