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、2位数字表示:所在省份的代码
+ * - 第3、4位数字表示:所在城市的代码
+ * - 第5、6位数字表示:所在区县的代码
+ * - 第7~14位数字表示:出生年、月、日
+ * - 第15、16位数字表示:所在地的派出所的代码
+ * - 第17位数字表示性别:奇数表示男性,偶数表示女性
+ * - 第18位数字是校检码,用来检验身份证的正确性。校检码可以是0~9的数字,有时也用x表示
+ *
+ *
+ * 第十八位数字(校验码)的计算方法为:
+ *
+ * - 将前面的身份证号码17位数分别乘以不同的系数。从第一位到第十七位的系数分别为:7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
+ * - 将这17位数字和系数相乘的结果相加
+ * - 用加出来和除以11,看余数是多少
+ * - 余数只可能有0 1 2 3 4 5 6 7 8 9 10这11个数字。其分别对应的最后一位身份证的号码为1 0 X 9 8 7 6 5 4 3 2
+ * - 通过上面得知如果余数是2,就会在身份证的第18位数字上出现罗马数字的Ⅹ。如果余数是10,身份证的最后一位号码就是2
+ *
+ *
+ * @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 + "]";
- }
}
}