Prepare release

This commit is contained in:
Looly 2025-02-18 09:19:56 +08:00
commit 21d3dadd13
55 changed files with 1013 additions and 95 deletions

View File

@ -1,6 +1,22 @@
# 🚀Changelog
-------------------------------------------------------------------------------------------------------------
# 5.8.36(2025-02-18)
### 🐣新特性
* 【crypto 】 增加BCUtil.decodeECPrivateKey方法issue#3829@Github
* 【core 】 增加HtmlUtil.cleanEmptyTag方法pr#3838@Github
* 【db 】 GlobalDbSetting优化默认配置读取规则优先读取文件而非jar中的文件issue#900@Github
* 【dfa 】 删除StopChar类中存在重复字符pr#3841@Github
* 【http 】 支持鸿蒙设备 UA 解析pr#1301@Gitee
### 🐞Bug修复
* 【aop 】 修复ProxyUtil可能的空指针问题issue#IBF20Z@Gitee
* 【core 】 修复XmlUtil转义调用方法错误问题修复XmlEscape未转义单引号问题pr#3837@Github
* 【core 】 修复FileUtil.isAbsolutePath没有判断smb路径问题pr#1299@Gitee
* 【core 】 修复AbstractFilter没有检查参数长度问题issue#3854@Github
-------------------------------------------------------------------------------------------------------------
# 5.8.35(2024-12-25)

View File

@ -153,18 +153,18 @@ We provide the T-Shirt and Sweater with Hutool Logo, please visit the shop
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</dependency>
```
### 🍐Gradle
```
implementation 'cn.hutool:hutool-all:5.8.35'
implementation 'cn.hutool:hutool-all:5.8.36'
```
## 📥Download
- [Maven Repo](https://repo1.maven.org/maven2/cn/hutool/hutool-all/5.8.35/)
- [Maven Repo](https://repo1.maven.org/maven2/cn/hutool/hutool-all/5.8.36/)
> 🔔note:
> Hutool 5.x supports JDK8+ and is not tested on Android platforms, and cannot guarantee that all tool classes or tool methods are available.

View File

@ -146,20 +146,20 @@ Hutool = Hu + tool是原公司项目底层代码剥离后的开源库“Hu
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</dependency>
```
### 🍐Gradle
```
implementation 'cn.hutool:hutool-all:5.8.35'
implementation 'cn.hutool:hutool-all:5.8.36'
```
### 📥下载jar
点击以下链接,下载`hutool-all-X.X.X.jar`即可:
- [Maven中央库](https://repo1.maven.org/maven2/cn/hutool/hutool-all/5.8.35/)
- [Maven中央库](https://repo1.maven.org/maven2/cn/hutool/hutool-all/5.8.36/)
> 🔔️注意
> Hutool 5.x支持JDK8+对Android平台没有测试不能保证所有工具类或工具方法可用。

View File

@ -4,3 +4,8 @@
# 多模块聚合文档生成在target/site/apidocs
exec mvn javadoc:aggregate
bin_home="$(dirname ${BASH_SOURCE[0]})"
# 拷贝自定义的index.html到聚合文档目录
cp -vf $bin_home/../docs/apidocs/index.html $bin_home/../target/reports/apidocs/

View File

@ -1 +1 @@
5.8.35
5.8.36

View File

@ -1 +1 @@
var version = '5.8.35'
var version = '5.8.36'

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-all</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-aop</artifactId>

View File

@ -12,6 +12,11 @@ import cn.hutool.aop.interceptor.JdkInterceptor;
public class JdkProxyFactory extends ProxyFactory {
private static final long serialVersionUID = 1L;
/**
* 获取单例
*/
public static JdkProxyFactory INSTANCE = new JdkProxyFactory();
@Override
public <T> T proxy(T target, Aspect aspect) {
return ProxyUtil.newProxyInstance(//

View File

@ -59,7 +59,13 @@ public abstract class ProxyFactory implements Serializable {
* @return 代理对象
*/
public static <T> T createProxy(T target, Aspect aspect) {
return create().proxy(target, aspect);
ProxyFactory factory = create();
if(null == factory){
// issue#IBF20Z
// 可能的空指针问题
factory = JdkProxyFactory.INSTANCE;
}
return factory.proxy(target, aspect);
}
/**

View File

@ -0,0 +1,49 @@
package cn.hutool.aop.test;
import cn.hutool.aop.proxy.ProxyFactory;
import cn.hutool.core.lang.Console;
import cn.hutool.core.thread.ThreadUtil;
import org.junit.jupiter.api.Test;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.atomic.AtomicInteger;
import static org.junit.jupiter.api.Assertions.assertEquals;
public class IssueIBF20ZTest {
@Test
public void testLoadFirstAvailableConcurrent() throws InterruptedException {
// 创建一个固定大小的线程池
int threadCount = 1000;
ExecutorService executorService = ThreadUtil.newExecutor(threadCount);
// 创建一个 CountDownLatch用于等待所有任务完成
CountDownLatch latch = new CountDownLatch(threadCount);
// 计数器用于统计成功加载服务提供者的次数
AtomicInteger successCount = new AtomicInteger(0);
// 提交多个任务到线程池
for (int i = 0; i < threadCount; i++) {
executorService.submit(() -> {
ProxyFactory factory = ProxyFactory.create();
if (factory != null) {
Console.log(factory.getClass());
successCount.incrementAndGet();
}
latch.countDown(); // 每个任务完成时计数减一
});
}
// 等待所有任务完成
latch.await();
// 关闭线程池并等待所有任务完成
executorService.shutdown();
// 验证所有线程都成功加载了服务提供者
assertEquals(threadCount, successCount.get());
}
}

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-bloomFilter</artifactId>

View File

@ -4,6 +4,7 @@ import cn.hutool.bloomfilter.BloomFilter;
import cn.hutool.bloomfilter.bitMap.BitMap;
import cn.hutool.bloomfilter.bitMap.IntMap;
import cn.hutool.bloomfilter.bitMap.LongMap;
import cn.hutool.core.lang.Assert;
/**
* 抽象Bloom过滤器
@ -46,7 +47,7 @@ public abstract class AbstractFilter implements BloomFilter {
* @param machineNum 机器位数
*/
public void init(long maxValue, int machineNum) {
this.size = maxValue;
this.size = Assert.checkBetween(maxValue, 1, Integer.MAX_VALUE);
switch (machineNum) {
case BitMap.MACHINE32:
bm = new IntMap((int) (size / machineNum));

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-bom</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-cache</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-captcha</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-core</artifactId>

View File

@ -64,6 +64,10 @@ public class FileUtil extends PathUtil {
*/
private static final Pattern PATTERN_PATH_ABSOLUTE = Pattern.compile("^[a-zA-Z]:([/\\\\].*)?", Pattern.DOTALL);
/**
* windows的共享文件夹开头
*/
private static final String SMB_PATH_PREFIX = "\\\\";
/**
* 是否为Windows环境
@ -1370,6 +1374,7 @@ public class FileUtil extends PathUtil {
* <li>/开头的路径</li>
* <li>满足类似于 c:/xxxxx其中祖母随意不区分大小写</li>
* <li>满足类似于 d:\xxxxx其中祖母随意不区分大小写</li>
* <li>满足windows SMB协议格式: \\192.168.254.1\Share</li>
* </ul>
*
* @param path 需要检查的Path
@ -1381,7 +1386,7 @@ public class FileUtil extends PathUtil {
}
// 给定的路径已经是绝对路径了
return StrUtil.C_SLASH == path.charAt(0) || ReUtil.isMatch(PATTERN_PATH_ABSOLUTE, path);
return StrUtil.C_SLASH == path.charAt(0) || path.startsWith(SMB_PATH_PREFIX) || ReUtil.isMatch(PATTERN_PATH_ABSOLUTE, path);
}
/**
@ -1667,7 +1672,7 @@ public class FileUtil extends PathUtil {
}
//兼容Windows下的共享目录路径原始路径如果以\\开头则保留这种路径
if (path.startsWith("\\\\")) {
if (path.startsWith(SMB_PATH_PREFIX)) {
return path;
}

View File

@ -79,6 +79,9 @@ public class FileResource implements Resource, Serializable {
@Override
public InputStream getStream() throws NoResourceException {
if (!this.file.exists()) {
throw new NoResourceException("File [{}] not exist!", this.file.getAbsolutePath());
}
return FileUtil.getInputStream(this.file);
}

View File

@ -1,6 +1,7 @@
package cn.hutool.core.text.escape;
import cn.hutool.core.text.replacer.LookupReplacer;
import cn.hutool.core.text.replacer.ReplacerChain;
/**
* HTML4的ESCAPE
@ -9,9 +10,21 @@ import cn.hutool.core.text.replacer.LookupReplacer;
* @author looly
*
*/
public class Html4Escape extends XmlEscape {
public class Html4Escape extends ReplacerChain {
private static final long serialVersionUID = 1L;
/**
* HTML转义字符<br>
* HTML转义相比XML并不转义单引号<br>
* https://stackoverflow.com/questions/1091945/what-characters-do-i-need-to-escape-in-xml-documents
*/
protected static final String[][] BASIC_ESCAPE = { //
{"\"", "&quot;"}, // " - double-quote
{"&", "&amp;"}, // & - ampersand
{"<", "&lt;"}, // < - less-than
{">", "&gt;"}, // > - greater-than
};
protected static final String[][] ISO8859_1_ESCAPE = { //
{ "\u00A0", "&nbsp;" }, // non-breaking space
{ "\u00A1", "&iexcl;" }, // inverted exclamation mark
@ -310,6 +323,7 @@ public class Html4Escape extends XmlEscape {
public Html4Escape() {
super();
addChain(new LookupReplacer(BASIC_ESCAPE));
addChain(new LookupReplacer(ISO8859_1_ESCAPE));
addChain(new LookupReplacer(HTML40_EXTENDED_ESCAPE));
}

View File

@ -22,7 +22,7 @@ public class XmlEscape extends ReplacerChain {
private static final long serialVersionUID = 1L;
protected static final String[][] BASIC_ESCAPE = { //
// {"'", "&apos;"}, // " - single-quote
{"'", "&apos;"}, // " - single-quote
{"\"", "&quot;"}, // " - double-quote
{"&", "&amp;"}, // & - ampersand
{"<", "&lt;"}, // < - less-than

View File

@ -13,8 +13,6 @@ public class XmlUnescape extends ReplacerChain {
private static final long serialVersionUID = 1L;
protected static final String[][] BASIC_UNESCAPE = InternalEscapeUtil.invert(XmlEscape.BASIC_ESCAPE);
// issue#1118
protected static final String[][] OTHER_UNESCAPE = new String[][]{new String[]{"&apos;", "'"}};
/**
* 构造
@ -22,6 +20,5 @@ public class XmlUnescape extends ReplacerChain {
public XmlUnescape() {
addChain(new LookupReplacer(BASIC_UNESCAPE));
addChain(new NumericEntityUnescaper());
addChain(new LookupReplacer(OTHER_UNESCAPE));
}
}

View File

@ -956,6 +956,7 @@ public class XmlUtil {
* &lt; (小于) 替换为 &amp;lt;
* &gt; (大于) 替换为 &amp;gt;
* &quot; (双引号) 替换为 &amp;quot;
* ' (单引号) 替换为 &amp;apos;
* </pre>
*
* @param string 被替换的字符串
@ -963,7 +964,7 @@ public class XmlUtil {
* @since 4.0.8
*/
public static String escape(String string) {
return EscapeUtil.escapeHtml4(string);
return EscapeUtil.escapeXml(string);
}
/**
@ -975,7 +976,7 @@ public class XmlUtil {
* @since 5.0.6
*/
public static String unescape(String string) {
return EscapeUtil.unescapeHtml4(string);
return EscapeUtil.unescapeXml(string);
}
/**

View File

@ -0,0 +1,61 @@
package cn.hutool.core.bean;
import cn.hutool.core.bean.copier.CopyOptions;
import cn.hutool.core.convert.TypeConverter;
import cn.hutool.core.date.DateUtil;
import cn.hutool.core.util.ReflectUtil;
import lombok.Data;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
import java.lang.reflect.Field;
import java.util.Date;
/**
* 自定义某个字段的转换
*/
public class IssueIBLTZWTest {
@Test
public void copyTest() {
TestBean bean = new TestBean();
bean.setName("test");
bean.setDate(DateUtil.parse("2025-02-17"));
final TestBean2 testBean2 = new TestBean2();
BeanUtil.copyProperties(bean, testBean2, createCopyOptions(TestBean2.class));
Assertions.assertEquals("2025", testBean2.getDate());
}
@Data
static class TestBean {
private String name;
private Date date;
}
@Data
static class TestBean2 {
private String name;
private String date;
}
static CopyOptions createCopyOptions(Class<?> targetClass) {
CopyOptions copyOptions = CopyOptions.create();
TypeConverter converter = (TypeConverter) ReflectUtil.getFieldValue(copyOptions, "converter");
copyOptions
.setIgnoreError(true) // 忽略类型错误避免自动转换
.setConverter(null)
.setFieldValueEditor((fieldName, value) -> {
try {
Field targetField = targetClass.getDeclaredField(fieldName);
// Date类型的 value instanceof 结果是String
if (targetField.getType() == String.class && value instanceof Date) {
return DateUtil.format((Date)value, "yyyy");
}
return converter.convert(targetField.getType(), value);
} catch (NoSuchFieldException e) {
return value;
}
});
return copyOptions;
}
}

View File

@ -53,6 +53,16 @@ public class FileUtilTest {
path = FileUtil.getAbsolutePath("d:");
assertEquals("d:", path);
}
@Test
public void smbPathTest() {
final String smbPath = "\\\\192.168.1.1\\share\\rc-source";
final String parseSmbPath = FileUtil.getAbsolutePath(smbPath);
assertEquals(smbPath, parseSmbPath);
assertTrue(FileUtil.isAbsolutePath(smbPath));
assertTrue(Paths.get(smbPath).isAbsolute());
}
@Test

View File

@ -0,0 +1,23 @@
package cn.hutool.core.io;
import cn.hutool.core.date.DateUtil;
import cn.hutool.core.date.StopWatch;
import cn.hutool.core.io.resource.ResourceUtil;
import cn.hutool.core.lang.Console;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Test;
import java.util.concurrent.TimeUnit;
public class Issue3846Test {
@Test
@Disabled
void readBytesTest() {
final StopWatch stopWatch = DateUtil.createStopWatch();
stopWatch.start();
final String filePath = "d:/test/issue3846.data";
final byte[] bytes = IoUtil.readBytes(ResourceUtil.getStream(filePath), false);
stopWatch.stop();
Console.log(stopWatch.prettyPrint(TimeUnit.MILLISECONDS));
}
}

View File

@ -3,12 +3,15 @@ package cn.hutool.core.map;
import cn.hutool.core.convert.Convert;
import cn.hutool.core.lang.Dict;
import cn.hutool.core.lang.Opt;
import cn.hutool.core.lang.TypeReference;
import cn.hutool.core.util.StrUtil;
import lombok.Builder;
import lombok.Data;
import org.junit.jupiter.api.Test;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;
@ -61,10 +64,10 @@ public class MapUtilTest {
public void mapTest() {
// Add test like a foreigner
final Map<Integer, String> adjectivesMap = MapUtil.<Integer, String>builder()
.put(0, "lovely")
.put(1, "friendly")
.put(2, "happily")
.build();
.put(0, "lovely")
.put(1, "friendly")
.put(2, "happily")
.build();
final Map<Integer, String> resultMap = MapUtil.map(adjectivesMap, (k, v) -> v + " " + PeopleEnum.values()[k].name().toLowerCase());
@ -80,7 +83,7 @@ public class MapUtilTest {
final Map<Long, User> idUserMap = Stream.iterate(0L, i -> ++i).limit(4).map(i -> User.builder().id(i).name(customers.poll()).build()).collect(Collectors.toMap(User::getId, Function.identity()));
// 如你所见它是一个mapkey由分组idvalue由用户ids组成典型的多对多关系
final Map<Long, List<Long>> groupIdUserIdsMap = groups.stream().flatMap(group -> idUserMap.keySet().stream().map(userId -> UserGroup.builder().groupId(group.getId()).userId(userId).build()))
.collect(Collectors.groupingBy(UserGroup::getGroupId, Collectors.mapping(UserGroup::getUserId, Collectors.toList())));
.collect(Collectors.groupingBy(UserGroup::getGroupId, Collectors.mapping(UserGroup::getUserId, Collectors.toList())));
// 神奇的魔法发生了 分组id和用户ids组成的map竟然变成了订单编号和用户实体集合组成的map
final Map<Long, List<User>> groupIdUserMap = MapUtil.map(groupIdUserIdsMap, (groupId, userIds) -> userIds.stream().map(idUserMap::get).collect(Collectors.toList()));
@ -190,11 +193,11 @@ public class MapUtilTest {
}
@Test
public void sortJoinTest(){
public void sortJoinTest() {
final Map<String, String> build = MapUtil.builder(new HashMap<String, String>())
.put("key1", "value1")
.put("key3", "value3")
.put("key2", "value2").build();
.put("key1", "value1")
.put("key3", "value3")
.put("key2", "value2").build();
final String join1 = MapUtil.sortJoin(build, StrUtil.EMPTY, StrUtil.EMPTY, false);
assertEquals("key1value1key2value2key3value3", join1);
@ -207,7 +210,7 @@ public class MapUtilTest {
}
@Test
public void ofEntriesTest(){
public void ofEntriesTest() {
final Map<String, Integer> map = MapUtil.ofEntries(MapUtil.entry("a", 1), MapUtil.entry("b", 2));
assertEquals(2, map.size());
@ -216,7 +219,19 @@ public class MapUtilTest {
}
@Test
public void getIntTest(){
public void ofEntriesSimpleEntryTest() {
final Map<String, Integer> map = MapUtil.ofEntries(
MapUtil.entry("a", 1, false),
MapUtil.entry("b", 2, false)
);
assertEquals(2, map.size());
assertEquals(Integer.valueOf(1), map.get("a"));
assertEquals(Integer.valueOf(2), map.get("b"));
}
@Test
public void getIntTest() {
assertThrows(NumberFormatException.class, () -> {
final HashMap<String, String> map = MapUtil.of("age", "d");
final Integer age = MapUtil.getInt(map, "age");
@ -238,18 +253,613 @@ public class MapUtilTest {
assertEquals("张三", map.get("newName"));
}
@Test
public void renameKeyMapEmptyNoChange() {
Map<String, String> map = new HashMap<>();
Map<String, String> result = MapUtil.renameKey(map, "oldKey", "newKey");
assertTrue(result.isEmpty());
}
@Test
public void renameKeyOldKeyNotPresentNoChange() {
Map<String, String> map = new HashMap<>();
map.put("anotherKey", "value");
Map<String, String> result = MapUtil.renameKey(map, "oldKey", "newKey");
assertEquals(1, result.size());
assertEquals("value", result.get("anotherKey"));
}
@Test
public void renameKeyOldKeyPresentNewKeyNotPresentKeyRenamed() {
Map<String, String> map = new HashMap<>();
map.put("oldKey", "value");
Map<String, String> result = MapUtil.renameKey(map, "oldKey", "newKey");
assertEquals(1, result.size());
assertEquals("value", result.get("newKey"));
}
@Test
public void renameKeyNewKeyPresentThrowsException() {
Map<String, String> map = new HashMap<>();
map.put("oldKey", "value");
map.put("newKey", "existingValue");
assertThrows(IllegalArgumentException.class, () -> {
MapUtil.renameKey(map, "oldKey", "newKey");
});
}
@Test
public void issue3162Test() {
final Map<String, Object> map = new HashMap<String, Object>() {
private static final long serialVersionUID = 1L;
{
put("a", "1");
put("b", "2");
put("c", "3");
}};
put("a", "1");
put("b", "2");
put("c", "3");
}
};
final Map<String, Object> filtered = MapUtil.filter(map, "a", "b");
assertEquals(2, filtered.size());
assertEquals("1", filtered.get("a"));
assertEquals("2", filtered.get("b"));
}
@Test
public void partitionNullMapThrowsException() {
assertThrows(IllegalArgumentException.class, () -> MapUtil.partition(null, 2));
}
@Test
public void partitionSizeZeroThrowsException() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
assertThrows(IllegalArgumentException.class, () -> MapUtil.partition(map, 0));
}
@Test
public void partitionSizeNegativeThrowsException() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
assertThrows(IllegalArgumentException.class, () -> MapUtil.partition(map, -1));
}
@Test
public void partitionEmptyMapReturnsEmptyList() {
Map<String, String> map = new HashMap<>();
List<Map<String, String>> result = MapUtil.partition(map, 2);
assertTrue(result.isEmpty());
}
@Test
public void partitionMapSizeMultipleOfSizePartitionsCorrectly() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
map.put("d", "4");
List<Map<String, String>> result = MapUtil.partition(map, 2);
assertEquals(2, result.size());
assertEquals(2, result.get(0).size());
assertEquals(2, result.get(1).size());
}
@Test
public void partitionMapSizeNotMultipleOfSizePartitionsCorrectly() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
map.put("d", "4");
map.put("e", "5");
List<Map<String, String>> result = MapUtil.partition(map, 2);
assertEquals(3, result.size());
assertEquals(2, result.get(0).size());
assertEquals(2, result.get(1).size());
assertEquals(1, result.get(2).size());
}
@Test
public void partitionGeneralCasePartitionsCorrectly() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
map.put("d", "4");
map.put("e", "5");
map.put("f", "6");
List<Map<String, String>> result = MapUtil.partition(map, 3);
assertEquals(2, result.size());
assertEquals(3, result.get(0).size());
assertEquals(3, result.get(1).size());
}
// ---------MapUtil.computeIfAbsentForJdk8
@Test
public void computeIfAbsentForJdk8KeyExistsReturnsExistingValue() {
Map<String, Integer> map = new HashMap<>();
map.put("key", 10);
Integer result = MapUtil.computeIfAbsentForJdk8(map, "key", k -> 20);
assertEquals(10, result);
}
@Test
public void computeIfAbsentForJdk8KeyDoesNotExistComputesAndInsertsValue() {
Map<String, Integer> map = new HashMap<>();
Integer result = MapUtil.computeIfAbsentForJdk8(map, "key", k -> 20);
assertEquals(20, result);
assertEquals(20, map.get("key"));
}
@Test
public void computeIfAbsentForJdk8ConcurrentInsertReturnsOldValue() {
ConcurrentHashMap<String, Integer> concurrentMap = new ConcurrentHashMap<>();
concurrentMap.put("key", 30);
AtomicInteger counter = new AtomicInteger(0);
// 模拟并发插入
concurrentMap.computeIfAbsent("key", k -> {
counter.incrementAndGet();
return 40;
});
Integer result = MapUtil.computeIfAbsentForJdk8(concurrentMap, "key", k -> 50);
assertEquals(30, result);
assertEquals(30, concurrentMap.get("key"));
assertEquals(0, counter.get());
}
@Test
public void computeIfAbsentForJdk8NullValueComputesAndInsertsValue() {
Map<String, Integer> map = new HashMap<>();
map.put("key", null);
Integer result = MapUtil.computeIfAbsentForJdk8(map, "key", k -> 20);
assertEquals(20, result);
assertEquals(20, map.get("key"));
}
//--------MapUtil.computeIfAbsent
@Test
public void computeIfAbsentKeyExistsReturnsExistingValue() {
Map<String, Integer> map = new HashMap<>();
map.put("key", 10);
Integer result = MapUtil.computeIfAbsent(map, "key", k -> 20);
assertEquals(10, result);
}
@Test
public void computeIfAbsentKeyDoesNotExistComputesAndInsertsValue() {
Map<String, Integer> map = new HashMap<>();
Integer result = MapUtil.computeIfAbsent(map, "key", k -> 20);
assertEquals(20, result);
assertEquals(20, map.get("key"));
}
@Test
public void computeIfAbsentConcurrentInsertReturnsOldValue() {
ConcurrentHashMap<String, Integer> concurrentMap = new ConcurrentHashMap<>();
concurrentMap.put("key", 30);
AtomicInteger counter = new AtomicInteger(0);
// 模拟并发插入
concurrentMap.computeIfAbsent("key", k -> {
counter.incrementAndGet();
return 40;
});
Integer result = MapUtil.computeIfAbsent(concurrentMap, "key", k -> 50);
assertEquals(30, result);
assertEquals(30, concurrentMap.get("key"));
assertEquals(0, counter.get());
}
@Test
public void computeIfAbsentNullValueComputesAndInsertsValue() {
Map<String, Integer> map = new HashMap<>();
map.put("key", null);
Integer result = MapUtil.computeIfAbsent(map, "key", k -> 20);
assertEquals(20, result);
assertEquals(20, map.get("key"));
}
@Test
public void computeIfAbsentEmptyMapInsertsValue() {
Map<String, Integer> map = new HashMap<>();
Integer result = MapUtil.computeIfAbsent(map, "newKey", k -> 100);
assertEquals(100, result);
assertEquals(100, map.get("newKey"));
}
@Test
public void computeIfAbsentJdk8KeyExistsReturnsExistingValue() {
Map<String, Integer> map = new HashMap<>();
// 假设JdkUtil.ISJDK8为true
map.put("key", 10);
Integer result = MapUtil.computeIfAbsent(map, "key", k -> 20);
assertEquals(10, result);
}
@Test
public void computeIfAbsentJdk8KeyDoesNotExistComputesAndInsertsValue() {
Map<String, Integer> map = new HashMap<>();
// 假设JdkUtil.ISJDK8为true
Integer result = MapUtil.computeIfAbsent(map, "key", k -> 20);
assertEquals(20, result);
assertEquals(20, map.get("key"));
}
//----------valuesOfKeys
@Test
public void valuesOfKeysEmptyIteratorReturnsEmptyList() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
Iterator<String> emptyIterator = Collections.emptyIterator();
ArrayList<String> result = MapUtil.valuesOfKeys(map, emptyIterator);
assertEquals(new ArrayList<String>(), result);
}
@Test
public void valuesOfKeysNonEmptyIteratorReturnsValuesList() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
Iterator<String> iterator = new ArrayList<String>() {
private static final long serialVersionUID = -4593258366224032110L;
{
add("a");
add("b");
}
}.iterator();
ArrayList<String> result = MapUtil.valuesOfKeys(map, iterator);
assertEquals(new ArrayList<String>() {
private static final long serialVersionUID = 7218152799308667271L;
{
add("1");
add("2");
}
}, result);
}
@Test
public void valuesOfKeysKeysNotInMapReturnsNulls() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
Iterator<String> iterator = new ArrayList<String>() {
private static final long serialVersionUID = -5479427021989481058L;
{
add("d");
add("e");
}
}.iterator();
ArrayList<String> result = MapUtil.valuesOfKeys(map, iterator);
assertEquals(new ArrayList<String>() {
private static final long serialVersionUID = 4390715387901549136L;
{
add(null);
add(null);
}
}, result);
}
@Test
public void valuesOfKeysMixedKeysReturnsMixedValues() {
Map<String, String> map = new HashMap<>();
map.put("a", "1");
map.put("b", "2");
map.put("c", "3");
Iterator<String> iterator = new ArrayList<String>() {
private static final long serialVersionUID = 8510595063492828968L;
{
add("a");
add("d");
add("b");
}
}.iterator();
ArrayList<String> result = MapUtil.valuesOfKeys(map, iterator);
assertEquals(new ArrayList<String>() {
private static final long serialVersionUID = 6383576410597048337L;
{
add("1");
add(null);
add("2");
}
}, result);
}
//--------clear
@Test
public void clearNoMapsProvidedNoAction() {
MapUtil.clear();
// 预期没有异常发生且没有Map被处理
}
@Test
public void clearEmptyMapNoChange() {
Map<String, String> map = new HashMap<>();
MapUtil.clear(map);
assertTrue(map.isEmpty());
}
@Test
public void clearNonEmptyMapClearsMap() {
Map<String, String> map = new HashMap<>();
map.put("key", "value");
MapUtil.clear(map);
assertTrue(map.isEmpty());
}
@Test
public void clearMultipleMapsClearsNonEmptyMaps() {
Map<String, String> map1 = new HashMap<>();
map1.put("key1", "value1");
Map<String, String> map2 = new HashMap<>();
map2.put("key2", "value2");
Map<String, String> map3 = new HashMap<>();
MapUtil.clear(map1, map2, map3);
assertTrue(map1.isEmpty());
assertTrue(map2.isEmpty());
assertTrue(map3.isEmpty());
}
@Test
public void clearMixedMapsClearsNonEmptyMaps() {
Map<String, String> map = new HashMap<>();
map.put("key", "value");
Map<String, String> emptyMap = new HashMap<>();
MapUtil.clear(map, emptyMap);
assertTrue(map.isEmpty());
assertTrue(emptyMap.isEmpty());
}
//-----empty
@Test
public void emptyNoParametersReturnsEmptyMap() {
Map<String, String> emptyMap = MapUtil.empty();
assertTrue(emptyMap.isEmpty(), "The map should be empty.");
assertSame(Collections.emptyMap(), emptyMap, "The map should be the same instance as Collections.emptyMap().");
}
@Test
public void emptyNullMapClassReturnsEmptyMap() {
Map<String, String> emptyMap = MapUtil.empty(null);
assertTrue(emptyMap.isEmpty(), "The map should be empty.");
assertSame(Collections.emptyMap(), emptyMap, "The map should be the same instance as Collections.emptyMap().");
}
@Test
public void emptyNavigableMapClassReturnsEmptyNavigableMap() {
Map<?, ?> map = MapUtil.empty(NavigableMap.class);
assertTrue(map.isEmpty());
assertInstanceOf(NavigableMap.class, map);
}
@Test
public void emptySortedMapClassReturnsEmptySortedMap() {
Map<?, ?> map = MapUtil.empty(SortedMap.class);
assertTrue(map.isEmpty());
assertInstanceOf(SortedMap.class, map);
}
@Test
public void emptyMapClassReturnsEmptyMap() {
Map<?, ?> map = MapUtil.empty(Map.class);
assertTrue(map.isEmpty());
}
@Test
public void emptyUnsupportedMapClassThrowsIllegalArgumentException() {
assertThrows(IllegalArgumentException.class, () -> {
MapUtil.empty(TreeMap.class);
});
}
//--------removeNullValue
@Test
public void removeNullValueNullMapReturnsNull() {
Map<String, String> result = MapUtil.removeNullValue(null);
assertNull(result);
}
@Test
public void removeNullValueEmptyMapReturnsEmptyMap() {
Map<String, String> map = new HashMap<>();
Map<String, String> result = MapUtil.removeNullValue(map);
assertEquals(0, result.size());
}
@Test
public void removeNullValueNoNullValuesReturnsSameMap() {
Map<String, String> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", "value2");
Map<String, String> result = MapUtil.removeNullValue(map);
assertEquals(2, result.size());
assertEquals("value1", result.get("key1"));
assertEquals("value2", result.get("key2"));
}
@Test
public void removeNullValueWithNullValuesRemovesNullEntries() {
Map<String, String> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", null);
map.put("key3", "value3");
Map<String, String> result = MapUtil.removeNullValue(map);
assertEquals(2, result.size());
assertEquals("value1", result.get("key1"));
assertEquals("value3", result.get("key3"));
assertNull(result.get("key2"));
}
@Test
public void removeNullValueAllNullValuesReturnsEmptyMap() {
Map<String, String> map = new HashMap<>();
map.put("key1", null);
map.put("key2", null);
Map<String, String> result = MapUtil.removeNullValue(map);
assertEquals(0, result.size());
}
//------getQuietly
@Test
public void getQuietlyMapIsNullReturnsDefaultValue() {
String result = MapUtil.getQuietly(null, "key1", new TypeReference<String>() {
}, "default");
assertEquals("default", result);
result = MapUtil.getQuietly(null, "key1", String.class, "default");
assertEquals("default", result);
}
@Test
public void getQuietlyKeyExistsReturnsConvertedValue() {
Map<String, Object> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", 123);
String result = MapUtil.getQuietly(map, "key1", new TypeReference<String>() {
}, "default");
assertEquals("value1", result);
}
@Test
public void getQuietlyKeyDoesNotExistReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", 123);
String result = MapUtil.getQuietly(map, "key3", new TypeReference<String>() {
}, "default");
assertEquals("default", result);
}
@Test
public void getQuietlyConversionFailsReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", 123);
Integer result = MapUtil.getQuietly(map, "key1", new TypeReference<Integer>() {
}, 0);
assertEquals(0, result);
}
@Test
public void getQuietlyKeyExistsWithCorrectTypeReturnsValue() {
Map<String, Object> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", 123);
Integer result = MapUtil.getQuietly(map, "key2", new TypeReference<Integer>() {
}, 0);
assertEquals(123, result);
}
@Test
public void getQuietlyKeyExistsWithNullValueReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("key1", "value1");
map.put("key2", 123);
map.put("key3", null);
String result = MapUtil.getQuietly(map, "key3", new TypeReference<String>() {
}, "default");
assertEquals("default", result);
}
@Test
public void getMapIsNullReturnsDefaultValue() {
assertNull(MapUtil.get(null, "age", String.class));
}
@Test
public void getKeyExistsReturnsConvertedValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals("18", MapUtil.get(map, "age", String.class));
}
@Test
public void getKeyDoesNotExistReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals("default", MapUtil.get(map, "nonexistent", String.class, "default"));
}
@Test
public void getTypeConversionFailsReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals(18, MapUtil.get(map, "age", Integer.class, 0));
}
@Test
public void getQuietlyTypeConversionFailsReturnsDefaultValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals(0, MapUtil.getQuietly(map, "name", Integer.class, 0));
}
@Test
public void getTypeReferenceReturnsConvertedValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals("18", MapUtil.get(map, "age", new TypeReference<String>() {
}));
}
@Test
public void getTypeReferenceWithDefaultValueReturnsConvertedValue() {
Map<String, Object> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals("18", MapUtil.get(map, "age", new TypeReference<String>() {
}, "default"));
}
@Test
public void getTypeReferenceWithDefaultValueTypeConversionFailsReturnsDefaultValue() {
Map<String, String> map = new HashMap<>();
map.put("age", "18");
map.put("name", "Hutool");
assertEquals(18, MapUtil.get(map, "age", new TypeReference<Integer>() {
}, 0));
map = null;
assertEquals(0, MapUtil.get(map, "age", new TypeReference<Integer>() {
}, 0));
}
}

View File

@ -7,7 +7,7 @@ import cn.hutool.core.lang.Console;
import cn.hutool.core.map.MapBuilder;
import cn.hutool.core.map.MapUtil;
import lombok.Data;
import static org.junit.jupiter.api.Assertions.*;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Test;
import org.w3c.dom.Document;
@ -22,6 +22,8 @@ import java.util.List;
import java.util.Map;
import java.util.Set;
import static org.junit.jupiter.api.Assertions.*;
/**
* {@link XmlUtil} 工具类
*
@ -319,7 +321,8 @@ public class XmlUtilTest {
public void escapeTest(){
final String a = "<>";
final String escape = XmlUtil.escape(a);
Console.log(escape);
Assertions.assertEquals("&lt;&gt;", escape);
Assertions.assertEquals("中文“双引号”", XmlUtil.escape("中文“双引号”"));
}
@Test

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-cron</artifactId>

View File

@ -17,7 +17,7 @@ public class CronTest {
@Test
@Disabled
public void customCronTest() {
CronUtil.schedule("*/2 * * * * *", (Task) () -> Console.log("Task excuted."));
CronUtil.schedule("*/2 * * * * *", (Task) () -> Console.log("Task executed."));
// 支持秒级别定时任务
CronUtil.setMatchSecond(true);

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-crypto</artifactId>

View File

@ -14,7 +14,9 @@ import org.bouncycastle.jcajce.provider.asymmetric.util.EC5Util;
import org.bouncycastle.jcajce.provider.asymmetric.util.ECUtil;
import org.bouncycastle.jce.spec.ECNamedCurveSpec;
import org.bouncycastle.jce.spec.ECParameterSpec;
import org.bouncycastle.jce.spec.ECPrivateKeySpec;
import org.bouncycastle.math.ec.ECCurve;
import org.bouncycastle.util.BigIntegers;
import java.io.IOException;
import java.io.InputStream;
@ -44,6 +46,27 @@ public class BCUtil {
return ((BCECPrivateKey) privateKey).getD().toByteArray();
}
/**
* 解码恢复EC私钥,支持Base64和Hex编码,基于BouncyCastle
*
* @param d 私钥d值
* @param curveName EC曲线名
* @return 私钥
* @since 5.8.36
*/
public static PrivateKey decodeECPrivateKey(final byte[] d, final String curveName) {
final X9ECParameters x9ECParameters = ECUtil.getNamedCurveByName(curveName);
final ECParameterSpec ecSpec = new ECParameterSpec(
x9ECParameters.getCurve(),
x9ECParameters.getG(),
x9ECParameters.getN(),
x9ECParameters.getH()
);
final ECPrivateKeySpec privateKeySpec = new ECPrivateKeySpec(BigIntegers.fromUnsignedByteArray(d), ecSpec);
return KeyUtil.generatePrivateKey("EC", privateKeySpec);
}
/**
* 编码压缩EC公钥基于BouncyCastle即Q值<br>
* https://www.cnblogs.com/xinzhao/p/8963724.html

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-db</artifactId>

View File

@ -1,6 +1,7 @@
package cn.hutool.db;
import cn.hutool.core.io.resource.NoResourceException;
import cn.hutool.core.util.ArrayUtil;
import cn.hutool.db.sql.SqlLog;
import cn.hutool.log.level.Level;
import cn.hutool.setting.Setting;
@ -83,20 +84,30 @@ public class GlobalDbConfig {
throw new NoResourceException("Customize db setting file [{}] not found !", dbSettingPath);
}
} else {
try {
setting = new Setting(DEFAULT_DB_SETTING_PATH, true);
} catch (NoResourceException e) {
// 尝试ClassPath下直接读取配置文件
try {
setting = new Setting(DEFAULT_DB_SETTING_PATH2, true);
} catch (NoResourceException e2) {
throw new NoResourceException("Default db setting [{}] or [{}] in classpath not found !", DEFAULT_DB_SETTING_PATH, DEFAULT_DB_SETTING_PATH2);
}
}
setting = tryDefaultDbSetting();
}
return setting;
}
/**
* 获取自定义或默认位置数据库配置{@link Setting}
*
* @return 数据库配置
* @since 5.8.36
*/
private static Setting tryDefaultDbSetting() {
final String[] defaultDbSettingPaths = {"file:" + DEFAULT_DB_SETTING_PATH, "file:" + DEFAULT_DB_SETTING_PATH2, DEFAULT_DB_SETTING_PATH, DEFAULT_DB_SETTING_PATH2};
for (final String settingPath : defaultDbSettingPaths) {
try {
return new Setting(settingPath, true);
} catch (final NoResourceException e) {
// ignore
}
}
throw new NoResourceException("Default db settings [{}] in classpath not found !", ArrayUtil.join(defaultDbSettingPaths, ","));
}
/**
* 设置全局配置是否通过debug日志显示SQL
*

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-dfa</artifactId>

View File

@ -1,37 +1,37 @@
package cn.hutool.dfa;
import java.util.Set;
import cn.hutool.core.collection.CollUtil;
import java.util.Set;
/**
* 过滤词及一些简单处理
*
*
* @author Looly
*/
public class StopChar {
/** 不需要处理的词,如标点符号、空格等 */
public static final Set<Character> STOP_WORD = CollUtil.newHashSet(' ', '\'', '、', '。', //
'·', 'ˉ', 'ˇ', '々', '—', '', '‖', '…', '', '', '“', '”', '', '', '〈', '〉', '《', '》', '「', '」', '『', //
'』', '〖', '〗', '【', '】', '±', '', '', '×', '÷', '∧', '', '∑', '∏', '', '∩', '∈', '√', '⊥', '⊙', '∫', //
'∮', '≡', '≌', '≈', '∽', '∝', '≠', '≮', '≯', '≤', '≥', '∞', '', '∵', '∴', '∷', '♂', '♀', '°', '', '〃', //
'℃', '', '¤', '¢', '£', '‰', '§', '☆', '★', '', '○', '●', '◎', '◇', '◆', '□', '■', '△', '▽', '⊿', '▲', //
'▼', '◣', '◤', '◢', '◥', '▁', '▂', '▃', '▄', '▅', '▆', '▇', '█', '▉', '▊', '▋', '▌', '▍', '▎', '▏', '▓', //
'※', '→', '←', '↑', '↓', '↖', '↗', '↘', '↙', '〓', '', 'ⅱ', 'ⅲ', 'ⅳ', '', 'ⅵ', 'ⅶ', 'ⅷ', 'ⅸ', '', '①', //
'②', '③', '④', '⑤', '⑥', '⑦', '⑧', '⑨', '⑩', '⒈', '⒉', '⒊', '⒋', '⒌', '⒍', '⒎', '⒏', '⒐', '⒑', '⒒', '⒓', //
'⒔', '⒕', '⒖', '⒗', '⒘', '⒙', '⒚', '⒛', '⑴', '⑵', '⑶', '⑷', '⑸', '⑹', '⑺', '⑻', '⑼', '⑽', '⑾', '⑿', '⒀', //
'⒁', '⒂', '⒃', '⒄', '⒅', '⒆', '⒇', '', 'Ⅱ', 'Ⅲ', 'Ⅳ', '', 'Ⅵ', 'Ⅶ', 'Ⅷ', 'Ⅸ', '', 'Ⅺ', 'Ⅻ', '', '”', //
'', '¥', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', //
'', '', '', '', '', '', '', '', '', '', '', '', '', '_', '', '', '', '', '∏', 'Ρ', '∑', //
'Υ', 'Φ', 'Χ', 'Ψ', 'Ω', 'α', 'β', 'γ', 'δ', 'ε', 'ζ', 'η', 'θ', 'ι', 'κ', 'λ', 'μ', 'ν', 'ξ', 'ο', 'π', //
'ρ', 'σ', 'τ', 'υ', 'φ', 'χ', 'ψ', 'ω', '', '', '', '', '', '﹊', '', '╭', '╮', '╰', '╯', '', '_', //
'', '^', '', '^', '', '', '/', '\\', '\"', '<', '>', '`', '·', '。', '{', '}', '~', '', '(', ')', '-', //
'√', '$', '@', '*', '&', '#', '卐', '㎎', '㎏', '㎜', '㎝', '㎞', '㎡', '㏄', '㏎', '㏑', '㏒', '㏕', '+', '=', '?',
':', '.', '!', ';', ']','|','%');
'·', 'ˉ', 'ˇ', '々', '—', '', '‖', '…', '', '', '“', '”', '', '', '〈', '〉', '《', '》', '「', '」', '『', //
'』', '〖', '〗', '【', '】', '±', '', '', '×', '÷', '∧', '', '∑', '∏', '', '∩', '∈', '√', '⊥', '⊙', '∫', //
'∮', '≡', '≌', '≈', '∽', '∝', '≠', '≮', '≯', '≤', '≥', '∞', '', '∵', '∴', '∷', '♂', '♀', '°', '', '〃', //
'℃', '', '¤', '¢', '£', '‰', '§', '☆', '★', '', '○', '●', '◎', '◇', '◆', '□', '■', '△', '▽', '⊿', '▲', //
'▼', '◣', '◤', '◢', '◥', '▁', '▂', '▃', '▄', '▅', '▆', '▇', '█', '▉', '▊', '▋', '▌', '▍', '▎', '▏', '▓', //
'※', '→', '←', '↑', '↓', '↖', '↗', '↘', '↙', '〓', '', 'ⅱ', 'ⅲ', 'ⅳ', '', 'ⅵ', 'ⅶ', 'ⅷ', 'ⅸ', '', '①', //
'②', '③', '④', '⑤', '⑥', '⑦', '⑧', '⑨', '⑩', '⒈', '⒉', '⒊', '⒋', '⒌', '⒍', '⒎', '⒏', '⒐', '⒑', '⒒', '⒓', //
'⒔', '⒕', '⒖', '⒗', '⒘', '⒙', '⒚', '⒛', '⑴', '⑵', '⑶', '⑷', '⑸', '⑹', '⑺', '⑻', '⑼', '⑽', '⑾', '⑿', '⒀', //
'⒁', '⒂', '⒃', '⒄', '⒅', '⒆', '⒇', '', 'Ⅱ', 'Ⅲ', 'Ⅳ', '', 'Ⅵ', 'Ⅶ', 'Ⅷ', 'Ⅸ', '', 'Ⅺ', 'Ⅻ', '', //
'', '¥', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', //
'', '', '', '', '', '', '', '', '', '', '', '', '', '_', '', '', '', '', 'Ρ', //
'Υ', 'Φ', 'Χ', 'Ψ', 'Ω', 'α', 'β', 'γ', 'δ', 'ε', 'ζ', 'η', 'θ', 'ι', 'κ', 'λ', 'μ', 'ν', 'ξ', 'ο', 'π', //
'ρ', 'σ', 'τ', 'υ', 'φ', 'χ', 'ψ', 'ω', '', '', '', '﹊', '', '╭', '╮', '╰', '╯', '', '_', //
'', '^', '', '', '', '/', '\\', '\"', '<', '>', '`', '{', '}', '~', '(', ')', '-', //
'$', '@', '*', '&', '#', '卐', '㎎', '㎏', '㎜', '㎝', '㎞', '㎡', '㏄', '㏎', '㏑', '㏒', '㏕', '+', '=', '?',
':', '.', '!', ';', ']','|','%');
/**
* 判断指定的词是否是不处理的词 如果参数为空则返回true因为空也属于不处理的字符
*
*
* @param ch 指定的词
* @return 是否是不处理的词
*/
@ -41,7 +41,7 @@ public class StopChar {
/**
* 是否为合法字符待处理字符
*
*
* @param ch 指定的词
* @return 是否为合法字符待处理字符
*/

View File

@ -1,11 +1,12 @@
package cn.hutool.dfa;
import cn.hutool.core.collection.CollUtil;
import static org.junit.jupiter.api.Assertions.*;
import org.junit.jupiter.api.Test;
import java.util.List;
import static org.junit.jupiter.api.Assertions.*;
/**
* DFA单元测试
*
@ -27,7 +28,7 @@ public class DfaTest {
// 匹配到就不再继续匹配了因此大土豆不匹配
// 匹配到刚出锅就跳过这三个字了因此出锅不匹配由于刚首先被匹配因此长的被匹配最短匹配只针对第一个字相同选最短
List<String> matchAll = tree.matchAll(text, -1, false, false);
assertEquals(matchAll, CollUtil.newArrayList("", "土^豆", "刚出锅"));
assertEquals(CollUtil.newArrayList("", "土^豆", "刚出锅"), matchAll);
}
/**

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-extra</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-http</artifactId>

View File

@ -24,6 +24,7 @@ public class HtmlUtil {
public static final String GT = StrUtil.HTML_GT;
public static final String RE_HTML_MARK = "(<[^<]*?>)|(<[\\s]*?/[^<]*?>)|(<[^<]*?/[\\s]*?>)";
public static final String RE_HTML_EMPTY_MARK = "<(\\w+)([^>]*)>\\s*</\\1>";
public static final String RE_SCRIPT = "<[\\s]*?script[^>]*?>.*?<[\\s]*?\\/[\\s]*?script[\\s]*?>";
private static final char[][] TEXT = new char[256][];
@ -86,6 +87,17 @@ public class HtmlUtil {
return content.replaceAll(RE_HTML_MARK, "");
}
/**
* 清除所有HTML空标签<br>
* 例如&lt;p&gt;&lt;/p&gt;
*
* @param content 文本
* @return 清除空标签后的文本
*/
public static String cleanEmptyTag(String content) {
return content.replaceAll(RE_HTML_EMPTY_MARK, "");
}
/**
* 清除指定HTML标签和被标签包围的内容<br>
* 不区分大小写

View File

@ -36,6 +36,7 @@ public class OS extends UserAgentInfo {
new OS("Windows", "windows"), //
new OS("OSX", "os x (\\d+)[._](\\d+)", "os x (\\d+([._]\\d+)*)"), //
new OS("Android", "Android", "Android (\\d+([._]\\d+)*)"),//
new OS("Harmony", "OpenHarmony", "OpenHarmony (\\d+([._]\\d+)*)"), //
new OS("Android", "XiaoMi|MI\\s+", "\\(X(\\d+([._]\\d+)*)"),//
new OS("Linux", "linux"), //
new OS("Wii", "wii", "wii libnup/(\\d+([._]\\d+)*)"), //

View File

@ -36,6 +36,11 @@ public class Platform extends UserAgentInfo {
* android
*/
public static final Platform ANDROID = new Platform("Android", "android");
/**
* harmony
*/
public static final Platform HARMONY = new Platform("Harmony", "OpenHarmony");
/**
* android
*/
@ -59,7 +64,8 @@ public class Platform extends UserAgentInfo {
GOOGLE_TV, //
new Platform("htcFlyer", "htc_flyer"), //
new Platform("Symbian", "symbian(os)?"), //
new Platform("Blackberry", "blackberry") //
new Platform("Blackberry", "blackberry"), //
HARMONY
);
/**
@ -144,4 +150,13 @@ public class Platform extends UserAgentInfo {
return this.equals(ANDROID) || this.equals(GOOGLE_TV);
}
/**
* 是否为Harmony平台
*
* @return 是否为Harmony平台
*/
public boolean isHarmony() {
return this.equals(HARMONY);
}
}

View File

@ -1,8 +1,9 @@
package cn.hutool.http;
import static org.junit.jupiter.api.Assertions.*;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;
/**
* Html单元测试
*
@ -77,6 +78,29 @@ public class HtmlUtilTest {
assertEquals("pre\r\n\t\tdfdsfdsfdsf\r\nBBBB", result);
}
@Test
public void cleanEmptyTagTest() {
String str = "<p></p><div></div>";
String result = HtmlUtil.cleanEmptyTag(str);
assertEquals("", result);
str = "<p>TEXT</p><div></div>";
result = HtmlUtil.cleanEmptyTag(str);
assertEquals("<p>TEXT</p>", result);
str = "<p></p><div>TEXT</div>";
result = HtmlUtil.cleanEmptyTag(str);
assertEquals("<div>TEXT</div>", result);
str = "<p>TEXT</p><div>TEXT</div>";
result = HtmlUtil.cleanEmptyTag(str);
assertEquals("<p>TEXT</p><div>TEXT</div>", result);
str = "TEXT<p></p><div></div>TEXT";
result = HtmlUtil.cleanEmptyTag(str);
assertEquals("TEXTTEXT", result);
}
@Test
public void unwrapHtmlTagTest() {
//非闭合标签

View File

@ -413,6 +413,16 @@ public class UserAgentUtilTest {
assertEquals("Linux", ua.getOs().toString());
}
@Test
public void parseHarmonyUATest() {
final String uaStr = "Mozilla/5.0 (Phone; OpenHarmony 4.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36 ArkWeb/4.1.6.1 Mobile ";
final UserAgent ua = UserAgentUtil.parse(uaStr);
assertEquals("Harmony", ua.getPlatform().toString());
assertTrue(ua.getPlatform().isHarmony());
assertEquals("Harmony", ua.getOs().toString());
assertEquals("4.1", ua.getOsVersion());
}
@Test
public void issueI60UOPTest() {
final String uaStr = "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36 dingtalk-win/1.0.0 nw(0.14.7) DingTalk(6.5.40-Release.9059101) Mojo/1.0.0 Native AppType(release) Channel/201200";

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-json</artifactId>

View File

@ -117,6 +117,7 @@ public interface JSON extends Cloneable, Serializable, IJSONTypeConverter {
*
* @param expression 表达式
* @param targetType 返回值类型
* @param <T> 获取对象类型
* @return 对象
* @see BeanPath#get(Object)
* @since 5.8.34

View File

@ -729,4 +729,15 @@ public class JSONObjectTest {
});
assertEquals("value2_edit", jsonObject.get("b"));
}
@Test
void issue3844Test(){
String camelCaseStr = "{\"userAge\":\"123\"}";
final JSONObject entries = new JSONObject(camelCaseStr, null, (pair) -> {
pair.setKey(StrUtil.toUnderlineCase(pair.getKey()));
return true;
});
Console.log(entries);
}
}

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-jwt</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-log</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-poi</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-script</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-setting</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-socket</artifactId>

View File

@ -9,7 +9,7 @@
<parent>
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
</parent>
<artifactId>hutool-system</artifactId>

View File

@ -8,7 +8,7 @@
<groupId>cn.hutool</groupId>
<artifactId>hutool-parent</artifactId>
<version>5.8.35</version>
<version>5.8.36</version>
<name>hutool</name>
<description>Hutool是一个小而全的Java工具类库通过静态方法封装降低相关API的学习成本提高工作效率使Java拥有函数式语言般的优雅让Java语言也可以“甜甜的”。</description>
<url>https://github.com/dromara/hutool</url>
@ -126,7 +126,7 @@
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-javadoc-plugin</artifactId>
<version>3.8.0</version>
<version>3.11.2</version>
<executions>
<execution>
<phase>package</phase>