From 6110a20f0a84e778d945c45fc5c9dd98370a9ee7 Mon Sep 17 00:00:00 2001 From: Looly Date: Sat, 2 Jan 2021 11:20:19 +0800 Subject: [PATCH] add method for #1348 --- CHANGELOG.md | 1 + .../java/cn/hutool/core/util/IdcardUtil.java | 50 ++++++++++++++++--- .../cn/hutool/core/util/IdcardUtilTest.java | 14 +++++- .../http/ssl/AndroidSupportSSLFactory.java | 15 +++--- .../java/cn/hutool/json/JSONArrayTest.java | 29 ++++------- 5 files changed, 76 insertions(+), 33 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d857c36b4..ca0a194e6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,7 @@ ### 新特性 * 【core 】 DynaBean.create增加重载方法(pr#245@Gitee) +* 【core 】 IdcardUtil增加重载是否忽略大小写(issue#1348@Github) ### Bug修复 diff --git a/hutool-core/src/main/java/cn/hutool/core/util/IdcardUtil.java b/hutool-core/src/main/java/cn/hutool/core/util/IdcardUtil.java index af9dc87de..e1bef751a 100644 --- a/hutool-core/src/main/java/cn/hutool/core/util/IdcardUtil.java +++ b/hutool-core/src/main/java/cn/hutool/core/util/IdcardUtil.java @@ -147,7 +147,7 @@ public class IdcardUtil { } /** - * 是否有效身份证号 + * 是否有效身份证号,忽略X的大小写 * * @param idCard 身份证号,支持18位、15位和港澳台的10位 * @return 是否有效 @@ -198,9 +198,46 @@ public class IdcardUtil { * * * @param idcard 待验证的身份证 - * @return 是否有效的18位身份证 + * @return 是否有效的18位身份证,忽略x的大小写 */ public static boolean isValidCard18(String idcard) { + return isValidCard18(idcard, true); + } + + /** + *

+ * 判断18位身份证的合法性 + *

+ * 根据〖中华人民共和国国家标准GB11643-1999〗中有关公民身份号码的规定,公民身份号码是特征组合码,由十七位数字本体码和一位数字校验码组成。
+ * 排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。 + *

+ * 顺序码: 表示在同一地址码所标识的区域范围内,对同年、同月、同 日出生的人编定的顺序号,顺序码的奇数分配给男性,偶数分配 给女性。 + *

+ *
    + *
  1. 第1、2位数字表示:所在省份的代码
  2. + *
  3. 第3、4位数字表示:所在城市的代码
  4. + *
  5. 第5、6位数字表示:所在区县的代码
  6. + *
  7. 第7~14位数字表示:出生年、月、日
  8. + *
  9. 第15、16位数字表示:所在地的派出所的代码
  10. + *
  11. 第17位数字表示性别:奇数表示男性,偶数表示女性
  12. + *
  13. 第18位数字是校检码,用来检验身份证的正确性。校检码可以是0~9的数字,有时也用x表示
  14. + *
+ *

+ * 第十八位数字(校验码)的计算方法为: + *

    + *
  1. 将前面的身份证号码17位数分别乘以不同的系数。从第一位到第十七位的系数分别为:7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
  2. + *
  3. 将这17位数字和系数相乘的结果相加
  4. + *
  5. 用加出来和除以11,看余数是多少
  6. + *
  7. 余数只可能有0 1 2 3 4 5 6 7 8 9 10这11个数字。其分别对应的最后一位身份证的号码为1 0 X 9 8 7 6 5 4 3 2
  8. + *
  9. 通过上面得知如果余数是2,就会在身份证的第18位数字上出现罗马数字的Ⅹ。如果余数是10,身份证的最后一位号码就是2
  10. + *
+ * + * @param idcard 待验证的身份证 + * @param ignoreCase 是否忽略大小写。{@code true}则忽略X大小写,否则严格匹配大写。 + * @return 是否有效的18位身份证 + * @since 5.5.7 + */ + public static boolean isValidCard18(String idcard, boolean ignoreCase) { if (CHINA_ID_MAX_LENGTH != idcard.length()) { return false; } @@ -217,13 +254,12 @@ public class IdcardUtil { } // 前17位 - String code17 = idcard.substring(0, 17); - // 第18位 - char code18 = Character.toLowerCase(idcard.charAt(17)); + final String code17 = idcard.substring(0, 17); if (ReUtil.isMatch(PatternPool.NUMBERS, code17)) { // 获取校验位 char val = getCheckCode18(code17); - return val == code18; + // 第18位 + return CharUtil.equals(val, idcard.charAt(17), ignoreCase); } return false; } @@ -586,7 +622,7 @@ public class IdcardUtil { case 3: return '9'; case 2: - return 'x'; + return 'X'; case 1: return '0'; case 0: diff --git a/hutool-core/src/test/java/cn/hutool/core/util/IdcardUtilTest.java b/hutool-core/src/test/java/cn/hutool/core/util/IdcardUtilTest.java index c25a9fdf8..4e3a85610 100644 --- a/hutool-core/src/test/java/cn/hutool/core/util/IdcardUtilTest.java +++ b/hutool-core/src/test/java/cn/hutool/core/util/IdcardUtilTest.java @@ -89,8 +89,20 @@ public class IdcardUtilTest { @Test public void isValidCard18Test(){ - final boolean isValidCard18 = IdcardUtil.isValidCard18("3301022011022000D6"); + boolean isValidCard18 = IdcardUtil.isValidCard18("3301022011022000D6"); Assert.assertFalse(isValidCard18); + + // 不忽略大小写情况下,X严格校验必须大写 + isValidCard18 = IdcardUtil.isValidCard18("33010219200403064x", false); + Assert.assertFalse(isValidCard18); + isValidCard18 = IdcardUtil.isValidCard18("33010219200403064X", false); + Assert.assertTrue(isValidCard18); + + // 非严格校验下大小写皆可 + isValidCard18 = IdcardUtil.isValidCard18("33010219200403064x"); + Assert.assertTrue(isValidCard18); + isValidCard18 = IdcardUtil.isValidCard18("33010219200403064X"); + Assert.assertTrue(isValidCard18); } @Test diff --git a/hutool-http/src/main/java/cn/hutool/http/ssl/AndroidSupportSSLFactory.java b/hutool-http/src/main/java/cn/hutool/http/ssl/AndroidSupportSSLFactory.java index a7a8cd6d9..fae1ddfcb 100644 --- a/hutool-http/src/main/java/cn/hutool/http/ssl/AndroidSupportSSLFactory.java +++ b/hutool-http/src/main/java/cn/hutool/http/ssl/AndroidSupportSSLFactory.java @@ -1,26 +1,27 @@ package cn.hutool.http.ssl; +import java.security.KeyManagementException; +import java.security.NoSuchAlgorithmException; + import static cn.hutool.http.ssl.SSLSocketFactoryBuilder.SSLv3; import static cn.hutool.http.ssl.SSLSocketFactoryBuilder.TLSv1; import static cn.hutool.http.ssl.SSLSocketFactoryBuilder.TLSv11; import static cn.hutool.http.ssl.SSLSocketFactoryBuilder.TLSv12; -import java.security.KeyManagementException; -import java.security.NoSuchAlgorithmException; - /** - * 兼容android低版本SSL连接 - * 咱在测试HttpUrlConnection的时候 - * 发现一部分手机无法连接[GithubPage] + * 兼容android低版本SSL连接
+ * 在测试HttpUrlConnection的时候,发现一部分手机无法连接[GithubPage] * + *

* 最后发现原来是某些SSL协议没有开启 + * * @author MikaGuraNTK */ public class AndroidSupportSSLFactory extends CustomProtocolsSSLFactory { // Android低版本不重置的话某些SSL访问就会失败 private static final String[] protocols = {SSLv3, TLSv1, TLSv11, TLSv12}; - + public AndroidSupportSSLFactory() throws KeyManagementException, NoSuchAlgorithmException { super(protocols); } diff --git a/hutool-json/src/test/java/cn/hutool/json/JSONArrayTest.java b/hutool-json/src/test/java/cn/hutool/json/JSONArrayTest.java index 5b42385e9..d4954a092 100644 --- a/hutool-json/src/test/java/cn/hutool/json/JSONArrayTest.java +++ b/hutool-json/src/test/java/cn/hutool/json/JSONArrayTest.java @@ -11,6 +11,7 @@ import cn.hutool.core.util.CharsetUtil; import cn.hutool.json.test.bean.Exam; import cn.hutool.json.test.bean.JsonNode; import cn.hutool.json.test.bean.KeyBean; +import lombok.Data; import org.junit.Assert; import org.junit.Ignore; import org.junit.Test; @@ -202,6 +203,14 @@ public class JSONArrayTest { Assert.assertEquals("-0", nodeList.get(3).getName()); } + @Test + public void getByPathTest(){ + String jsonStr = "[{\"id\": \"1\",\"name\": \"a\"},{\"id\": \"2\",\"name\": \"b\"}]"; + final JSONArray jsonArray = JSONUtil.parseArray(jsonStr); + Assert.assertEquals("b", jsonArray.getByPath("[1].name")); + Assert.assertEquals("b", JSONUtil.getByPath(jsonArray, "[1].name")); + } + private static Map buildMap(String id, String parentId, String name) { Map map = new HashMap<>(); map.put("id", id); @@ -209,26 +218,10 @@ public class JSONArrayTest { map.put("name", name); return map; } - + + @Data static class User { private Integer id; private String name; - - public Integer getId() { - return id; - } - public void setId(Integer id) { - this.id = id; - } - public String getName() { - return name; - } - public void setName(String name) { - this.name = name; - } - @Override - public String toString() { - return "User [id=" + id + ", name=" + name + "]"; - } } }