Java校验输入的身份证号

 在程序中我们总免不了对用户输入的数据进行上层校验,将格式不正确的数据直接拦截在调用整个方法之前,这样不仅能让用户很明确地知道自己输入的数据是否正确,错误的原因,还一定程度上提高了方法的效率。小编在最近的工作中,整理了关于身份证号的验证方式:

I 使用最朴实的Java代码实现

    我们都知道身份证号的组成受地区、出生年月的影响,所以工具类中自然少不了对这两个要素的判断:
// 地区集合
    final static Map<Integer, String> zoneNum = new HashMap<Integer, String>();
    static {
        zoneNum.put(11, "北京");
        zoneNum.put(12, "天津");
        zoneNum.put(13, "河北");
        zoneNum.put(14, "山西");
        zoneNum.put(15, "内蒙古");
        zoneNum.put(21, "辽宁");
        zoneNum.put(22, "吉林");
        zoneNum.put(23, "黑龙江");
        zoneNum.put(31, "上海");
        zoneNum.put(32, "江苏");
        zoneNum.put(33, "浙江");
        zoneNum.put(34, "安徽");
        zoneNum.put(35, "福建");
        zoneNum.put(36, "江西");
        zoneNum.put(37, "山东");
        zoneNum.put(41, "河南");
        zoneNum.put(42, "湖北");
        zoneNum.put(43, "湖南");
        zoneNum.put(44, "广东");
        zoneNum.put(45, "广西");
        zoneNum.put(46, "海南");
        zoneNum.put(50, "重庆");
        zoneNum.put(51, "四川");
        zoneNum.put(52, "贵州");
        zoneNum.put(53, "云南");
        zoneNum.put(54, "西藏");
        zoneNum.put(61, "陕西");
        zoneNum.put(62, "甘肃");
        zoneNum.put(63, "青海");
        zoneNum.put(64, "新疆");
        zoneNum.put(71, "台湾");
        zoneNum.put(81, "香港");
        zoneNum.put(82, "澳门");
        zoneNum.put(91, "外国");
    }
 
    final static int[] PARITYBIT = { '1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2' };
    final static int[] POWER_LIST = { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2 };
 
    /**
     * 身份证号是否合理的校验
     * 
     * @param certNo
     * @return
     */
    public static boolean isIDCard(String certNo) {
        if (certNo == null || (certNo.length() != 15 && certNo.length() != 18))
            return false;
        final char[] cs = certNo.toUpperCase().toCharArray();
        // 校验位数
        int power = 0;
        for (int i = 0; i < cs.length; i++) {
            if (i == cs.length - 1 && cs[i] == 'X')
                break;// 最后一位可以 是X或x
            if (cs[i] < '0' || cs[i] > '9')
                return false;
            if (i < cs.length - 1) {
                power += (cs[i] - '0') * POWER_LIST[i];
            }
        }
 
        // 校验区位码
        if (!zoneNum.containsKey(Integer.valueOf(certNo.substring(0, 2)))) {
            return false;
        }
 
        // 校验年份
        String year = certNo.length() == 15 ? getIdcardCalendar() + certNo.substring(6, 8) : certNo.substring(6, 10);
 
        final int iyear = Integer.parseInt(year);
        if (iyear < 1900 || iyear > Calendar.getInstance().get(Calendar.YEAR))
            return false;// 1900年的PASS,超过今年的PASS
 
        // 校验月份
        String month = certNo.length() == 15 ? certNo.substring(8, 10) : certNo.substring(10, 12);
        final int imonth = Integer.parseInt(month);
        if (imonth < 1 || imonth > 12) {
            return false;
        }
 
        // 校验天数
        String day = certNo.length() == 15 ? certNo.substring(10, 12) : certNo.substring(12, 14);
        final int iday = Integer.parseInt(day);
        if (iday < 1 || iday > 31)
            return false;
 
        // 校验"校验码"
        if (certNo.length() == 15)
            return true;
        return cs[cs.length - 1] == PARITYBIT[power % 11];
    }
 
    /**
     * 获取身份证号中的年月日
     * 
     * @return
     */
    private static int getIdcardCalendar() {
        GregorianCalendar curDay = new GregorianCalendar();
        int curYear = curDay.get(Calendar.YEAR);
        int year2bit = Integer.parseInt(String.valueOf(curYear).substring(2));
        return year2bit;
    }
      但是技术总是发展的,从某天开始,攻城狮们获得了一项福利--可以将上述繁杂的代码使用一个表达式来概括,那就是 正则表达式:

II  正则表达式:public static boolean verifyIdCard(String idCard){
        String strVerify = "(^[1-9]\\d{5}[1-9]\\d{3}((0\\d)|(1[0-2]))(([0|1|2]\\d)|3[0-1])\\d{3}(\\d|x|X)$)|(^[1-9]\\d{7}((0\\d)|(1[0-2]))(([0|1|2]\\d)|3[0-1])\\d{3}$)";
        Pattern p = Pattern.compile(strVerify);  
        Matcher m = p.matcher(idCard);
        return m.find();
    }
     在开发的工作中,我们应该尽量地提高代码的精炼程度,相比之下,还是比较倾向于正则表达式,毕竟两种方式的代码量差距实在太大了。

*************************************************************************************************************************************************

JAVA 判断身份证的正确性

/*********************************** 身份证验证开始 ****************************************/
/**
* 身份证号码验证 1、号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,
* 八位数字出生日期码,三位数字顺序码和一位数字校验码。 2、地址码(前六位数)
* 表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。 3、出生日期码(第七位至十四位)
* 表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。 4、顺序码(第十五位至十七位)
* 表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。 5、校验码(第十八位数)
* (1)十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, , 16 ,先对前17位数字的权求和
* Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
* 2 (2)计算模 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
*/
/**
* 功能:身份证的有效验证

* @param IDStr
*            身份证号
* @return 有效:返回"" 无效:返回String信息
* @throws ParseException
* @throws java.text.ParseException
* @throws NumberFormatException
*/
public static boolean IDCardValidate(String IDStr)
throws NumberFormatException, java.text.ParseException {
String errorInfo = "";// 记录错误信息
Log.e("IDStr", IDStr);
String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4",
"3", "2" };
String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7",
"9", "10", "5", "8", "4", "2" };
String Ai = "";
// ================ 号码的长度 15位或18位 ================
if (IDStr.length() != 15 && IDStr.length() != 18) {
errorInfo = "身份证号码长度应该为15位或18位。";
Log.e("errorInfo", errorInfo);
return false;
}
// =======================(end)========================
// ================ 数字 除最后以为都为数字 ================
if (IDStr.length() == 18) {
Ai = IDStr.substring(0, 17);
} else if (IDStr.length() == 15) {
Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
}
if (isNumeric(Ai) == false) {
errorInfo = "身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字。";
Log.e("errorInfo", errorInfo);
return false;
}
// =======================(end)========================
// ================ 出生年月是否有效 ================
Log.e("Ai", Ai);
String strYear = Ai.substring(6, 10);// 年份
String strMonth = Ai.substring(10, 12);// 月份
String strDay = Ai.substring(12, 14);// 月份
Log.e("strYear", strYear);
Log.e("strMonth", strMonth);
Log.e("strDay", strDay);
if (checkValidDate(strYear + strMonth + strDay) == false) {
Log.e("strYear + strMonth + strDay", strYear + strMonth + strDay);
errorInfo = "身份证生日无效。";
Log.e("errorInfo", errorInfo);
return false;
}
GregorianCalendar gc = new GregorianCalendar();
SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
|| (gc.getTime().getTime() - s.parse(
strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
errorInfo = "身份证生日不在有效范围。";
Log.e("errorInfo", errorInfo);
return false;
}
if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
errorInfo = "身份证月份无效";
Log.e("errorInfo", errorInfo);
return false;
}
if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
errorInfo = "身份证日期无效";
Log.e("errorInfo", errorInfo);
return false;
}
// =====================(end)=====================
// ================ 地区码时候有效 ================
Hashtable h = GetAreaCode();
if (h.get(Ai.substring(0, 2)) == null) {
errorInfo = "身份证地区编码错误。";
Log.e("errorInfo", errorInfo);
return false;
}
// ==============================================
// ================ 判断最后一位的值 ================
int TotalmulAiWi = 0;
for (int i = 0; i < 17; i++) {
TotalmulAiWi = TotalmulAiWi
+ Integer.parseInt(String.valueOf(Ai.charAt(i)))
* Integer.parseInt(Wi[i]);
}
int modValue = TotalmulAiWi % 11;
String strVerifyCode = ValCodeArr[modValue];
Ai = Ai + strVerifyCode;


if (IDStr.length() == 18) {
if (Ai.equals(IDStr) == false) {
errorInfo = "身份证无效,不是合法的身份证号码";
Log.e("errorInfo", errorInfo);
return false;
}
} else {
return true;
}
// =====================(end)=====================
return true;
}
/**
* 功能:设置地区编码

* @return Hashtable 对象
*/
@SuppressWarnings("unchecked")
private static Hashtable GetAreaCode() {
Hashtable hashtable = new Hashtable();
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;
}
/**
* 功能:判断字符串是否为数字

* @param str
* @return
*/
private static boolean isNumeric(String str) {
Pattern pattern = Pattern.compile("[0-9]*");
Matcher isNum = pattern.matcher(str);
if (isNum.matches()) {
return true;
} else {
return false;
}
}
/**
* 功能:判断字符串是否为日期格式

* @param str
* @return
*/


private static boolean checkValidDate(String pDateObj) {
boolean ret = true;
if (pDateObj == null || pDateObj.length() != 8) {
ret = false;
}
try {
int year = new Integer(pDateObj.substring(0, 4)).intValue();
int month = new Integer(pDateObj.substring(4, 6)).intValue();
int day = new Integer(pDateObj.substring(6)).intValue();
Calendar cal = Calendar.getInstance();
cal.setLenient(false); // 允许严格检查日期格式
cal.set(year, month - 1, day);
cal.getTime();// 该方法调用就会抛出异常
} catch (Exception e) {
ret = false;
}
return ret;
}
/*********************************** 身份证验证结束 ****************************************/
 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Sure, 我可以帮你解决这个问题。身份证号校验程序的实现可以通过以下步骤完成: 1. 在Java中创建一个新的类,使用Scanner类获取用户输入身份证号码。 2. 创建一个方法,该方法将输入身份证号码作为参数,并将其转换为char[]数组。 3. 对转换后的数组进行一些简单的验证,例如检查长度是否为18个字符以及前17个字符是否都是数字。 4. 使用第17个和第18个字符计算身份证号码的校验和。 5. 将ComputeCheckSum方法引用到ValidateID方法中,以计算并验证身份证号码的校验和。 6. 加入对地区代码和出生日期的验证。 以下是身份证号校验程序的示例代码: ``` import java.util.Scanner; public class ValidateID { public static void main(String[] args) { Scanner scanner = new Scanner(System.in); System.out.println("Please enter your ID number: "); String id = scanner.next(); if (ValidateID(id)) { System.out.println("Valid ID number"); } else { System.out.println("Invalid ID number"); } } public static boolean ValidateID(String id) { if (id == null || id.trim().length() != 18) { return false; } char[] ids = id.toCharArray(); for (int i = 0; i < ids.length - 1; i++) { if (!Character.isDigit(ids[i])) { return false; } } String[] wf = new String[]{"7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2"}; String[] vcode = new String[]{"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"}; int sum = 0; for (int i = 0; i < wf.length; i++) { sum += Integer.parseInt(wf[i]) * Integer.parseInt(String.valueOf(ids[i])); } int mod = sum % 11; String last = String.valueOf(ids[17]); if (mod == 2) { if (!last.equalsIgnoreCase("x")) { return false; } } else { if (!last.equals(vcode[mod])) { return false; } } String areaCode = id.substring(0, 6); String year = id.substring(6, 10); String month = id.substring(10, 12); String day = id.substring(12, 14); String birthday = year + "-" + month + "-" + day; boolean yearCheck = Integer.valueOf(year) >= 1900 && Integer.valueOf(year) <= (Calendar.getInstance().get(Calendar.YEAR) - 18); try { SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd"); format.setLenient(false); format.parse(birthday); } catch (Exception e) { return false; } boolean areaCheck = false; for (String area : areaCodeArray) { if (area.equals(areaCode.substring(0, 2))) { areaCheck = true; break; } } return yearCheck && areaCheck; } } ``` 请注意,我并不是储存已经输入的文本,所以我将无法记住之前提到的文本。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值