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();
}
}

JAVA验证身份证号码是否合法的更多相关文章

  1. JAVA验证身份证号码是否正确

    package com.IdCard; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.D ...

  2. java验证身份证号码是否有效源代码 wn25的头像 wn25 23 2015-01-04 20:09 6 基本信息 Java × 1 浏览

    原文:http://www.open-open.com/code/view/1420373343171 1.描述 用java语言判断身份证号码是否有效,地区码.出身年月.校验码等验证算法 2.源代码 ...

  3. java验证身份证号码是否有效源代码

    原文:http://www.open-open.com/code/view/1420373343171 1.描述 用java语言判断身份证号码是否有效,地区码.出身年月.校验码等验证算法 2.源代码 ...

  4. 如何用javascript正则表达式验证身份证号码是否合法

    在用户注册页面有些需求要求的比较严格,需要对身份证js验证是否合法,通过此功能严格此系统软件,从而过滤到很多水客.下面就此实现方法给大家讲解下. 很多时候我们都是通过一组正则表达式来判断用户输入的身份 ...

  5. JAVA通过身份证号码获取出生日期、年龄、性别

    JAVA验证身份证号码是否正确:https://www.cnblogs.com/pxblog/p/12038278.html /** * 通过身份证号码获取出生日期(birthday).年龄(age) ...

  6. 具体验证身份证号码规则和姓名(汉字)的java代码

    一.验证汉字的正則表達式  /** 是否是汉字的正则 */  private String regexIsHanZi = "[\\u4e00-\\u9fa5]+";   * @pa ...

  7. php验证身份证号码的正确性

    /********************php验证身份证号码是否正确函数*********************/function is_idcard( $id ) {   $id = strto ...

  8. C#验证身份证号码

    一.18位的身份证号码 如:130429####%%%%00781.1~6位为地区代码,其中1.2位数为各省级政府的代码,3.4位数为地.市级政府的代码,5.6位数为县.区级政府代码.如13(河北省) ...

  9. php验证身份证号码正确性

    发布:JB01   来源:脚本学堂     [大 中 小] 分享一例php代码,用于验证身份证号码的正确性,用到了preg_match.preg_replace函数,有需要的朋友可以参考学习下.本文转 ...

随机推荐

  1. angular之自定义管道

    1,装了angular2 的 cli之后,cmd中命令建立个管道文件 ng g p <name>; 如建一个在pipe文件中建一个add.pipe.ts文件 可以这么么写 ng g p p ...

  2. JSP 发送邮件

    JSP 发送邮件 虽然使用JSP实现邮件发送功能很简单,但是需要有JavaMail API,并且需要安装JavaBean Activation Framework. 在这里下载最新版本的 JavaMa ...

  3. JavaScript Browser 对象 实例

    使用JavaScript来访问和控制浏览器对象实例. Window 对象 弹出一个警告框 弹出一个带折行的警告框 弹出一个确认框,并提醒访客点击的内容 弹出一个提示框 点击一个按钮时,打开一个新窗口 ...

  4. Highcharts 基本区域图;Highcharts 使用负数区域图;Highcharts 堆叠区域图;Highcharts 百分比堆叠区域图

    Highcharts 基本区域图 配置 chart chart.type 配置项用于设定图表类型,默认为 "line",本章节我们使用 'area'. var chart = { ...

  5. JavaScript中的call、apply、bind是怎么回事?

    在JS中,这三者都是用来改变函数的this对象的指向的,他们有什么样的区别呢.在说区别之前还是先总结一下三者的相似之处:1.都是用来改变函数的this对象的指向的.2.第一个参数都是this要指向的对 ...

  6. C++进阶3.字节对齐 联合

    C++进阶3.字节对齐 联合 20131011 多益和金山笔试 知识漏洞 20131011 前言: 今天下午是多益网络的笔试,整体感觉还好,但是找到很多的知识漏洞.一直笔试到6:00,然后紧张的从会生 ...

  7. WebLogic和Tomcat

    J2ee开发主要是浏览器和服务器进行交互的一种结构.逻辑都是在后台进行处理,然后再把结果传输回给浏览器.可以看出服务器在这种架构是非常重要的. 这几天接触到两种Java的web服务器,做项目用的Tom ...

  8. MySQL mha 高可用集群搭建

    [mha] MHA作为MySQL故障切换和主从提升的高可用软件,在故障切换过程中,MHA能做到在0~30秒之内自动完成数据库的故障切换操作,并且在进行故障切换的过程中,MHA能在最大程度上保证数据的一 ...

  9. PCA--主成份分析

    主成份分析(Principle Component Analysis)主要用来对数据进行降维.对于高维数据,处理起来比较麻烦,而且高维数据可能含有相关的维度,数据存在冗余,PCA通过把高维数据向低维映 ...

  10. HashMap resize代码详解(二)

    关于其中的resize方法如下: final Node<K,V>[] resize() { Node<K,V>[] oldTab = table; int oldCap = ( ...