xml
<!-- 将引入Hutool的所有模块 -->
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.32</version>
</dependency>
<!-- Hutool核心模块 -->
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-core</artifactId>
<version>5.8.32</version>
</dependency>| 模块 | 介绍 |
|---|---|
| hutool-cache | 简单缓存实现 |
| hutool-core | Bean 操作、日期、各种 Util |
| hutool-crypto | 加密解密模块,提供对称、非对称和摘要算法封装 |
| hutool-dfa | 基于 DFA 模型的多关键字查找 |
| hutool-extra | 扩展模块,对第三方封装(模板引擎、邮件、Servlet、二维码、Emoji、FTP、分词等) |
| hutool-script | 脚本执行封装,例如 Javascript |
| hutool-system | 系统参数调用封装(JVM 信息等) |
| hutool-captcha | 图片验证码 |
| hutool-jwt | JWT 封装 |
❤ 工具类
💛 唯一键 IdUtil
UUID :全球唯一识别码
java
//生成的UUID是带-的字符串,类似于:a5c8a5e8-df2b-4706-bea4-08d0939410e3
String uuid = IdUtil.randomUUID();
//生成的是不带-的字符串,类似于:b17f24ff026d40949c85a24f4f375d42
String simpleUUID = IdUtil.simpleUUID();雪花算法 :分布式系统中,需要使用全局唯一 ID,这个 ID 要简单,但是又能够按照时间有序生成
- 有终端机器的使用
Snowflake getSnowflake(终端id,数据中心id)获取带有机器属性的 Snowflake 对象【范围是 0 - 31】long nextId()String nextIdStr()
- 简单使用
long getSnowflakeNextId()使用的是全局单例的雪花算法类,每秒最多能生成 4,096,000 个 IDString getSnowflakeNextIdStr()
java
// 有终端机器的使用
IdUtil.getSnowflake(23L, 110L).nextId()
// 简单使用
long id = IdUtil.getSnowflakeNextId();
String id = IdUtil.getSnowflakeNextIdStr();💛 随机 Random
RandomUtil静态对象randomInt()获取随机的 intrandomInt(int minInclude, int maxExclude)获得指定范围内的随机数(包左不包右)randomString(int length)获得一个随机的 length 长度的字符串
java
// 生成一个8位的随机数
String randomNumber = RandomUtil.randomString(8);
System.out.println("生成的8位随机数是:" + randomNumber);💙 权重随机
根据对象的权重值,来 Random 对象
java
// 1.构建权重对象WeightObj,将其加入list集合
List<WeightRandom.WeightObj<String>> weightList = Stream.of(
// 第一个值是权重对象,第二个值是权重
new WeightRandom.WeightObj<String>("A", 20),
new WeightRandom.WeightObj<String>("B", 30),
new WeightRandom.WeightObj<String>("C", 40),
new WeightRandom.WeightObj<String>("D", 10)
).collect(Collectors.toList());
// 2.生成权重随机对象
WeightRandom<String> wr = RandomUtil.weightRandom(weightList);
int numA = 0, numB = 0, numC = 0, numD = 0;
// 3.打印随机结果
for (int i = 0; i < 1000; i++) {
switch (wr.next()) {
case "A":
numA++;
break;
case "B":
numB++;
break;
case "C":
numC++;
break;
case "D":
numD++;
break;
}
}
System.out.println("A:" + numA);
System.out.println("B:" + numB);
System.out.println("C:" + numC);
System.out.println("D:" + numD);
---
A:210
B:315
C:375
D:100💛 Base64
[!hint] 在浏览器中打开 Base64 编码的图片 直接在浏览器的地址栏中输入:
 base64 编码数据
- 字节数组 -> base64
java
byte[] byteArray = new byte[10];
String base64 = Base64.encode(byteArray);💛 二维码 QrCodeUtil
[!hint] 使用之前还要再引入 zxing 依赖
xml\<dependency> \<groupId>com.google.zxing\</groupId> \<artifactId>core\</artifactId> \<version>3.5.3\</version> \</dependency>
💙 生成二维码
- 自定义二维码参数
QrConfig【长,宽,二维码的颜色,背景颜色,边距 ……】,这个参数可以作为后续生成二维码
java
// 设置长宽
QrConfig config = new QrConfig(300, 300);
// 设置边距,即二维码和背景之间的边距
config.setMargin(3);
// 设置前景色,即二维码颜色(青色)
config.setForeColor(Color.CYAN.getRGB());
// 附带logo
config.setImg("logo_small.jpg")- URL -> 二维码图片
java
QrCodeUtil.generate("https://hutool.cn/", 300, 300, FileUtil.file("d:/qrcode.jpg"));- URL ->
BufferedImage图片对象
java
String url = "https://www.baidu.com";
// 设置二维码的宽和高
QrConfig config = new QrConfig(300, 300);
// 生成二维码图片
BufferedImage qrImage = QrCodeUtil.generate(url, config);- URL ->
byte[]格式的 PNG 二维码图片
java
byte[] imageBytes = QrCodeUtil.generatePng("https://www.baidu.com", 300, 300);
String base64 = Base64.encode(imageBytes);
return base64;💙 识别二维码
java
// decode 就是被识别出来的链接
String decode = QrCodeUtil.decode(FileUtil.file("d:/qrcode.jpg"));💛 数字工具 NumberUtil
计算 :
- 加减乘除 :支持任意数字的包装类,String
NumberUtil.add加NumberUtil.sub减NumberUtil.mul乘NumberUtil.div除double div(double d, double dd, int scale)使用 d 除以 dd ,保留 scale 位小数double div(double d, flost ff, int scale)使用 d 除以 ff ,保留 scale 位小数
- 其他
multiple(int m, int n)最小公倍数
保留小数 :
BigDecimal round(数字n,精度s)保留数字 n 的后 s 位小数,支持 BigDecimal,Double,StringString roundStr(n, s)返回的是数字的字符串类型
校验数字 :
NumberUtil.isNumber是否为数字NumberUtil.isInteger是否为整数NumberUtil.isDouble是否为浮点数NumberUtil.isPrimes是否为质数
💛 字符串 StrUtil
判空 :
单个
isBlank()/isNotBlank()isEmpty()/isNotEmpty()containsAny(字符串s,待比较字符串ss ……)判断 s 是否在字符串 ss …… 中
集合
hasBlank(字符串数组)判断这个数组中是否有至少一个为 blank,如果是,则返回 truehasEmpty()
比较
equalsAny(s1, s2, s3 ……)判断 s1 是否等于后续的字符串中的任意一个
编辑 :
removeSuffix(字符串,字符)去除这个字符串最后一个该字符
💛 旧 - 日期时间
范围 :
rangeToList(开始时间,结束时间,单位)获取到两个时间之间的时间集合
java
List<DateTime> dateTimes = DateUtil.rangeToList(
DateUtil.parse("2024-03-01"),
DateUtil.parse("2024-05-01"),
DateField.DAY_OF_YEAR
);
dateTimes.forEach(System.out::println);💛 新 - 日期时间
格式化,解析 :
LocalDateTime parse(时间字符串,时间格式)字符串 -> LocalDateTime
java
LocalDateTime localDateTime2 = LocalDateTimeUtil.parse("2020-01-23 12:23:56", DatePattern.NORM_DATETIME_FORMATTER);String format(时间,时间格式)LocalDateTime -> 字符串
java
LocalDateTimeUtil.format(LocalDateTimeUtil.beginOfDay(LocalDateTime.parse("2024-09-01T00:00:00")), DatePattern.NORM_DATETIME_PATTERN)偏移时间量 :
offset(时间,偏移量,偏移单位)
java
LocalDateTime localDateTime = LocalDateTime.of(2021, 1, 1, 0, 0, 30);
LocalDateTime offset = LocalDateTimeUtil.offset(
localDateTime,
-30,
ChronoUnit.MINUTES
);
System.out.println(offset);
---
2020-12-31T23:30:30开始 / 结束 :
LocalDateTime beginOfDay(时间)例如 "2020-01-23T00:00"LocalDateTime endOfDay(时间)例如 "2020-01-23T23:59:59.999999999"
时间间隔 :
Duration between(开始时间,结束时间)计算时间差值
java
LocalDateTime start = LocalDateTime.now();
LocalDateTime end = LocalDateTime.now()
.plusDays(1)
.plusHours(33);
Duration between = LocalDateTimeUtil.between(start, end);
System.out.println(between.toHours());
System.out.println(between.toMinutes());
System.out.println(between.toSeconds());
System.out.println(between.toMillis());
---
57
3420
205200
205200000判断范围 :
isIn(需要判断的时间,开始时间点,结束时间点,是否包含开始时间,是否包含结束时间)判断某个时间是否在某个范围中isSameDay(日期1,日期2)比较两个时间是否是同一天,支持 LocalDateTime,LocalDateisWeekend(时间)判断这一天是否为周末,支持 LocalDateTime,LocalDate
java
boolean in = LocalDateTimeUtil.isIn(
LocalDateTime.now(),
LocalDateTime.now().minusDays(1),
LocalDateTime.now().plusDays(1),
true,
true
);❤️ 集合
流式 Map 创建
java
Map<String, Integer> map = MapBuilder.create(new HashMap<String, Integer>())
.put("a", 1)
.put("b", 2)
.map();CollUtil
判空 :
isEmpty()isNotEmpty()
ListUtil
分割集合 :
List<集合A> ListUtil.partition(集合A, 多少为一组);将集合 A 分割成以 n 为一组的集合
java
// finalUserIds有108个元素,将拆成3个finalUserIds,分别有50,50,8
List<List<String>> partition = ListUtil.partition(finalUserIds, 50);❤️ 数据结构
💛 树结构 TreeUtil
TreeUtil 用于构建复杂的菜单层级结构数据

Tree :虽然是一整棵树,但是当使用某些方法时,判断的逻辑是这棵树的根节点
- 【增】
addChildren(Tree ……)在 Tree 上加 TreesetChildren(List<Tree<T>> children)直接设置覆盖当前 Tree 节点的所有子节点
- 【改】
filter(Filter函数式接口)只将 Tree 的子节点进行过滤,Tree 的根节点会被保留。如果节点返回 false,则移除该节点及其所有子节点;如果节点返回 true,则移除该节点及其所有子节点。会直接修改原有的树,不需要返回值filterNew(Filter函数式接口)与 filter 相同,但是不会修改原有的树,而是返回新树walk(Consumer函数式接口)将该 Tree 的所有节点逐层依次遍历消费
- 【查】
getNode(子节点id)获取到该 Tree 下指定子节点 id 的子节点List<Tree<T>> getChildren()获取该 Tree 下的所有子节点树hasChild()判断是否有子节点getName()获取该 Tree 的当前节点的 name- ……
TreeNode :所有的方法都是简单的围绕属性值的 get,set 方法
- 【增】
new TreeNode<>(id, 父id, 节点名, 排序)排序值越小越靠前 - 【改】
- 【查】
getId()getName()获取该 Tree 的当前节点的 name- ……
简单示例 :
java
// 构建node列表
List<TreeNode<String>> springList = List.of(
// id, 父id, 节点名, 排序
new TreeNode<>("1", "0", "Spring", 5),
new TreeNode<>("11", "1", "Spring Boot", 222222),
new TreeNode<>("111", "11", "ORM 框架", 0),
new TreeNode<>("1111", "1", "Spring Data Jpa", 1)
);
// 构建树,并指定节点列表中的顶级节点id
List<Tree<String>> springTreeList = TreeUtil.build(springList, "0");
for (Tree<String> tree : springTreeList) {
System.out.println(tree.getName());
System.out.println(tree.getNode("11"));
}
System.out.println("====================================");
List<TreeNode<String>> springSecurityList = List.of(
new TreeNode<>("11111", "1", "Spring Security", 44),
new TreeNode<>("111111", "11111", "JWT", 2),
new TreeNode<>("1111111", "11111", "OAuth2", 3)
);
List<Tree<String>> springSecurityTreeList = TreeUtil.build(springSecurityList, "1");
springTreeList.stream()
.peek(tree -> tree.addChildren(springSecurityTreeList.get(0)))
.forEach(System.out::println);
---
Spring
Spring Boot[11]
ORM 框架[111]
====================================
Spring[1]
Spring Data Jpa[1111]
Spring Boot[11]
ORM 框架[111]
Spring Security[11111]
JWT[111111]
OAuth2[1111111]自定义字段 :扩展字段,自定义字段名
java
// 构建node列表
List<TreeNode<String>> nodeList = ……;
// 构建配置
TreeNodeConfig treeNodeConfig = new TreeNodeConfig()
.setIdKey("rid") // 设置ID的键名
.setParentIdKey("pid") // 设置父ID的键名
.setWeightKey("weight") // 设置权重的键名
.setNameKey("name") // 设置节点名的键名
.setDeep(3);// 设置树的最大深度
List<Tree<String>> treeNodes = TreeUtil.build(nodeList, "0", treeNodeConfig, (treeNode, tree) -> {
tree.setId(treeNode.getId());
tree.setParentId(treeNode.getParentId());
tree.setWeight(treeNode.getWeight());
tree.setName(treeNode.getName());
// 扩展属性
tree.putExtra("ext1", 1);
});
List<ObjectNode> list = treeNodes.stream()
.map(item -> new ObjectMapper().convertValue(item, ObjectNode.class))
.toList();
System.out.println(list);
---
[
{
"rid":"2",
"pid":"0",
"weight":1,
"name":"Go",
"ext1":1,
"children":[……]
},
……
]树形结构转 JSON :
java
List<Tree<String>> springTreeList = TreeUtil.build(springList, "0");
ObjectNode objectNode = new ObjectMapper().createObjectNode();
springTreeList.forEach(item -> objectNode.setAll(new ObjectMapper().convertValue(item, ObjectNode.class)));
System.out.println(objectNode.toPrettyString());
---
json 树形结构❤ Bean
Bean 就是有 setter,getter 的 Java 类
💛 Bean -> Bean
- Award 转 AwardBO
java
Award award = Award.builder()
.strategyId(1)
.awardId(101)
.awardKey("random_points")
.awardTitle("随机积分")
.awardCount(100)
.awardRate(0.1f)
.awardSort(1)
.rules("rules")
.build();
AwardBO awardBO = new AwardBO();
BeanUtil.copyProperties(award, awardBO);
System.out.println(awardBO);
---
AwardBO(strategyId=1, awardId=101, awardCount=100, awardRate=0.1)- Award 对象集合转 AwardBO 对象集合
java
List<Award> awards = Stream.of(
Award.builder()
.id(1).strategyId(1).awardId(101)
.awardKey("random").awardConfig("random")
.awardTitle("随机积分").awardSubtitle("副标题")
.awardCount(100).awardRate(0.1f).awardSort(1)
.rules("rules").createTime(LocalDateTime.now())
.updateTime(LocalDateTime.now())
.build(),
Award.builder()
.id(2).strategyId(1).awardId(102)
.awardKey("random").awardConfig("config")
.awardTitle("随机积分").awardSubtitle("副")
.awardCount(100).awardRate(0.1f)
.awardSort(1).rules("rules")
.createTime(LocalDateTime.now())
.updateTime(LocalDateTime.now())
.build()
).toList();
List<AwardBO> awardBOs = BeanUtil.copyToList(awards, AwardBO.class);
awardBOs.forEach(System.out::println);
---
AwardBO(strategyId=1, awardId=101, awardCount=100, awardRate=0.1)
AwardBO(strategyId=1, awardId=102, awardCount=100, awardRate=0.1)❤ 加密解密
[!quote] 加密分类
- 对称加密
- AES
- ARCFOUR
- Blowfish
- DES
- DESede
- RC2
- PBEWithMD5AndDES
- PBEWithSHA1AndDESede
- PBEWithSHA1AndRC2_40
- 非对称加密
- RSA
- DSA
- 摘要加密
- MD2
- MD5
- SHA-1
- SHA-256
- SHA-384
- SHA-512
- HmacMD5
- HmacSHA1
- HmacSHA256
- HmacSHA384
- HmacSHA512
常见加密 SecureUtil
- 对称加密
SecureUtil.aesSecureUtil.des
- 摘要加密
String SecureUtil.md5SecureUtil.md5()SecureUtil.md5(File dataFile)加密文件,生成 16 进制 MD5 字符串SecureUtil.md5(InputStream data)SecureUtil.md5(String data)
SecureUtil.sha1SecureUtil.hmacSecureUtil.hmacMd5SecureUtil.hmacSha1
- 非对称加密
SecureUtil.rsaSecureUtil.dsa
- UUID
SecureUtil.simpleUUID方法提供无“-”的 UUID
- 密钥生成
SecureUtil.generateKey针对对称加密生成密钥SecureUtil.generateKeyPair生成密钥对(用于非对称加密)SecureUtil.generateSignature生成签名(用于非对称加密)
java
String stringSignTemp = "AAA";
String MD5 = SecureUtil.md5(stringSignTemp);❤️ 网络 NetUtil
getLocalMacAddress()获取本机的 mac 地址
❤️ 缓存 CacheUtil
Cache 有五个实现类:
newTimedCachenewFIFOCachenewLRUCachenewWeakCachenewLFUCache
java
// 定时缓存:过期时间为3秒
Cache<Object, Object> timedCache = CacheUtil.newTimedCache(3000);
timedCache.put("key1", "value1");
System.out.println(timedCache.get("key1"));
// FIFO缓存:先进先出缓存,超出容量后先加入的缓存失效
Cache<String, String> fifoCache = CacheUtil.newFIFOCache(5);
// LRU缓存:超出容量时,会移除最近最少使用的缓存项
Cache<String, String> lruCache = CacheUtil.newLRUCache(5);
// 弱引用缓存:当没有指向缓存值的引用时,缓存值会被垃圾回收。也可以指定过期时间
Cache<String, String> weakCache = CacheUtil.newWeakCache(5000);
// LFU缓存:优先淘汰访问频率最低的缓存项
Cache<String, String> staticCache = CacheUtil.newLFUCache(100);java
// 手动封装全局缓存
public class PicPathGlobalCache {
// key 是唯一id,value 是图片路径列表
private static final Cache<String, List<String>> cache = CacheUtil.newFIFOCache(10);
public static void put(String key, List<String> val) {
cache.put(key, val);
}
public static List<String> get(String key) {
return cache.get(key);
}
}❤ 断言 Assert
[!hint] Java 原生的 Assert 的缺点 断言 本意上是在调式时使用,而不是生产环境
- assert 默认在程序运行时是关闭的,不会产生任何效果,只有在调试时会生效
- assert 断言失败,程序将退出
而 Hutool 的断言工具是用来做参数校验的
- 判空
isNull/notNull是否为 nullnotEmpty不能为空,支持字符串,数组,集合 ……notBlank不能是空白字符串
- 判对错
isTrue必须为 true,否则抛出 IllegalArgumentException 异常state检查 Boolean 表达式,如果为 false ,则抛异常
- 判包含
notContain不能包含指定的子串noNullElements数组中不能包含 null 元素
- 判类关系
isInstanceOf必须是指定类的实例isAssignable必须是子类和父类关系