mirror of
https://gitee.com/dromara/hutool.git
synced 2025-04-05 17:37:59 +08:00
add NanoId
This commit is contained in:
parent
6ebd912062
commit
f9b29c5f2a
@ -8,6 +8,7 @@
|
||||
### 🐣新特性
|
||||
* 【core 】 DateUtil增加ceiling重载,可选是否归零毫秒
|
||||
* 【core 】 IterUtil增加firstMatch方法
|
||||
* 【core 】 增加NanoId
|
||||
|
||||
### 🐞Bug修复
|
||||
* 【core 】 修复FileUtil.normalize处理上级路径的问题(issue#I3YPEH@Gitee)
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,9 +1,11 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
||||
|
||||
<parent>
|
||||
<groupId>cn.hutool</groupId>
|
||||
<artifactId>hutool-parent</artifactId>
|
||||
@ -13,7 +15,7 @@
|
||||
<artifactId>hutool-bloomFilter</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool 布隆过滤器</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,5 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>pom</packaging>
|
||||
|
@ -1,9 +1,11 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
||||
|
||||
<parent>
|
||||
<groupId>cn.hutool</groupId>
|
||||
<artifactId>hutool-parent</artifactId>
|
||||
@ -13,7 +15,7 @@
|
||||
<artifactId>hutool-cache</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool 缓存</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,9 +1,11 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
||||
|
||||
<parent>
|
||||
<groupId>cn.hutool</groupId>
|
||||
<artifactId>hutool-parent</artifactId>
|
||||
@ -13,7 +15,7 @@
|
||||
<artifactId>hutool-captcha</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool 验证码工具</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,104 +0,0 @@
|
||||
package cn.hutool.core.lang;
|
||||
|
||||
import java.security.SecureRandom;
|
||||
import java.util.Random;
|
||||
|
||||
/**
|
||||
* A class for generating unique String IDs.
|
||||
*
|
||||
* The implementations of the core logic in this class are based on NanoId, a JavaScript
|
||||
* library by Andrey Sitnik released under the MIT license. (https://github.com/ai/nanoid)
|
||||
*
|
||||
* @author David Klebanoff
|
||||
*/
|
||||
public final class NanoId {
|
||||
|
||||
/**
|
||||
* <code>NanoIdUtils</code> instances should NOT be constructed in standard programming.
|
||||
* Instead, the class should be used as <code>NanoIdUtils.randomNanoId();</code>.
|
||||
*/
|
||||
private NanoId() {
|
||||
//Do Nothing
|
||||
}
|
||||
|
||||
/**
|
||||
* The default random number generator used by this class.
|
||||
* Creates cryptographically strong NanoId Strings.
|
||||
*/
|
||||
public static final SecureRandom DEFAULT_NUMBER_GENERATOR = new SecureRandom();
|
||||
|
||||
/**
|
||||
* The default alphabet used by this class.
|
||||
* Creates url-friendly NanoId Strings using 64 unique symbols.
|
||||
*/
|
||||
public static final char[] DEFAULT_ALPHABET =
|
||||
"_-0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ".toCharArray();
|
||||
|
||||
/**
|
||||
* The default size used by this class.
|
||||
* Creates NanoId Strings with slightly more unique values than UUID v4.
|
||||
*/
|
||||
public static final int DEFAULT_SIZE = 21;
|
||||
|
||||
/**
|
||||
* Static factory to retrieve a url-friendly, pseudo randomly generated, NanoId String.
|
||||
*
|
||||
* The generated NanoId String will have 21 symbols.
|
||||
*
|
||||
* The NanoId String is generated using a cryptographically strong pseudo random number
|
||||
* generator.
|
||||
*
|
||||
* @return A randomly generated NanoId String.
|
||||
*/
|
||||
public static String randomNanoId() {
|
||||
return randomNanoId(DEFAULT_NUMBER_GENERATOR, DEFAULT_ALPHABET, DEFAULT_SIZE);
|
||||
}
|
||||
|
||||
/**
|
||||
* Static factory to retrieve a NanoId String.
|
||||
*
|
||||
* The string is generated using the given random number generator.
|
||||
*
|
||||
* @param random The random number generator.
|
||||
* @param alphabet The symbols used in the NanoId String.
|
||||
* @param size The number of symbols in the NanoId String.
|
||||
* @return A randomly generated NanoId String.
|
||||
*/
|
||||
public static String randomNanoId(final Random random, final char[] alphabet, final int size) {
|
||||
|
||||
if (random == null) {
|
||||
throw new IllegalArgumentException("random cannot be null.");
|
||||
}
|
||||
|
||||
if (alphabet == null) {
|
||||
throw new IllegalArgumentException("alphabet cannot be null.");
|
||||
}
|
||||
|
||||
if (alphabet.length == 0 || alphabet.length >= 256) {
|
||||
throw new IllegalArgumentException("alphabet must contain between 1 and 255 symbols.");
|
||||
}
|
||||
|
||||
if (size <= 0) {
|
||||
throw new IllegalArgumentException("size must be greater than zero.");
|
||||
}
|
||||
|
||||
final int mask = (2 << (int) Math.floor(Math.log(alphabet.length - 1) / Math.log(2))) - 1;
|
||||
final int step = (int) Math.ceil(1.6 * mask * size / alphabet.length);
|
||||
|
||||
final StringBuilder idBuilder = new StringBuilder();
|
||||
|
||||
while (true) {
|
||||
final byte[] bytes = new byte[step];
|
||||
random.nextBytes(bytes);
|
||||
for (int i = 0; i < step; i++) {
|
||||
final int alphabetIndex = bytes[i] & mask;
|
||||
if (alphabetIndex < alphabet.length) {
|
||||
idBuilder.append(alphabet[alphabetIndex]);
|
||||
if (idBuilder.length() == size) {
|
||||
return idBuilder.toString();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
103
hutool-core/src/main/java/cn/hutool/core/lang/id/NanoId.java
Normal file
103
hutool-core/src/main/java/cn/hutool/core/lang/id/NanoId.java
Normal file
@ -0,0 +1,103 @@
|
||||
package cn.hutool.core.lang.id;
|
||||
|
||||
import cn.hutool.core.util.RandomUtil;
|
||||
|
||||
import java.security.SecureRandom;
|
||||
import java.util.Random;
|
||||
|
||||
/**
|
||||
* NanoId,一个小型、安全、对 URL友好的唯一字符串 ID 生成器,特点:
|
||||
*
|
||||
* <ul>
|
||||
* <li>安全:它使用加密、强大的随机 API,并保证符号的正确分配</li>
|
||||
* <li>体积小:只有 258 bytes 大小(压缩后)、无依赖</li>
|
||||
* <li>紧凑:它使用比 UUID (A-Za-z0-9_~)更多的符号</li>
|
||||
* </ul>
|
||||
*
|
||||
* <p>
|
||||
* 此实现的逻辑基于JavaScript的NanoId实现,见:https://github.com/ai/nanoid
|
||||
*
|
||||
* @author David Klebanoff
|
||||
*/
|
||||
public class NanoId {
|
||||
|
||||
/**
|
||||
* 默认随机数生成器,使用{@link SecureRandom}确保健壮性
|
||||
*/
|
||||
private static final SecureRandom DEFAULT_NUMBER_GENERATOR = RandomUtil.getSecureRandom();
|
||||
|
||||
/**
|
||||
* 默认随机字母表,使用URL安全的Base64字符
|
||||
*/
|
||||
private static final char[] DEFAULT_ALPHABET =
|
||||
"_-0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ".toCharArray();
|
||||
|
||||
/**
|
||||
* 默认长度
|
||||
*/
|
||||
public static final int DEFAULT_SIZE = 21;
|
||||
|
||||
/**
|
||||
* 生成伪随机的NanoId字符串,长度为默认的{@link #DEFAULT_SIZE},使用密码安全的伪随机生成器
|
||||
*
|
||||
* @return 伪随机的NanoId字符串
|
||||
*/
|
||||
public static String randomNanoId() {
|
||||
return randomNanoId(DEFAULT_SIZE);
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成伪随机的NanoId字符串
|
||||
*
|
||||
* @param size ID长度
|
||||
* @return 伪随机的NanoId字符串
|
||||
*/
|
||||
public static String randomNanoId(int size) {
|
||||
return randomNanoId(null, null, size);
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成伪随机的NanoId字符串
|
||||
*
|
||||
* @param random 随机数生成器
|
||||
* @param alphabet 随机字母表
|
||||
* @param size ID长度
|
||||
* @return 伪随机的NanoId字符串
|
||||
*/
|
||||
public static String randomNanoId(Random random, char[] alphabet, int size) {
|
||||
if (random == null) {
|
||||
random = DEFAULT_NUMBER_GENERATOR;
|
||||
}
|
||||
|
||||
if (alphabet == null) {
|
||||
alphabet = DEFAULT_ALPHABET;
|
||||
}
|
||||
|
||||
if (alphabet.length == 0 || alphabet.length >= 256) {
|
||||
throw new IllegalArgumentException("Alphabet must contain between 1 and 255 symbols.");
|
||||
}
|
||||
|
||||
if (size <= 0) {
|
||||
throw new IllegalArgumentException("Size must be greater than zero.");
|
||||
}
|
||||
|
||||
final int mask = (2 << (int) Math.floor(Math.log(alphabet.length - 1) / Math.log(2))) - 1;
|
||||
final int step = (int) Math.ceil(1.6 * mask * size / alphabet.length);
|
||||
|
||||
final StringBuilder idBuilder = new StringBuilder();
|
||||
|
||||
while (true) {
|
||||
final byte[] bytes = new byte[step];
|
||||
random.nextBytes(bytes);
|
||||
for (int i = 0; i < step; i++) {
|
||||
final int alphabetIndex = bytes[i] & mask;
|
||||
if (alphabetIndex < alphabet.length) {
|
||||
idBuilder.append(alphabet[alphabetIndex]);
|
||||
if (idBuilder.length() == size) {
|
||||
return idBuilder.toString();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
7
hutool-core/src/main/java/cn/hutool/core/lang/id/package-info.java
Executable file
7
hutool-core/src/main/java/cn/hutool/core/lang/id/package-info.java
Executable file
@ -0,0 +1,7 @@
|
||||
/**
|
||||
* 提供各种ID生成
|
||||
*
|
||||
* @author looly
|
||||
* @since 5.7.5
|
||||
*/
|
||||
package cn.hutool.core.lang.id;
|
@ -1,14 +1,13 @@
|
||||
package cn.hutool.core.util;
|
||||
|
||||
import cn.hutool.core.exceptions.UtilException;
|
||||
import cn.hutool.core.lang.*;
|
||||
import cn.hutool.core.lang.ObjectId;
|
||||
import cn.hutool.core.lang.Singleton;
|
||||
import cn.hutool.core.lang.Snowflake;
|
||||
import cn.hutool.core.lang.UUID;
|
||||
import cn.hutool.core.lang.id.NanoId;
|
||||
import cn.hutool.core.net.NetUtil;
|
||||
|
||||
import java.util.Random;
|
||||
|
||||
import static cn.hutool.core.lang.NanoId.DEFAULT_ALPHABET;
|
||||
import static cn.hutool.core.lang.NanoId.DEFAULT_NUMBER_GENERATOR;
|
||||
|
||||
/**
|
||||
* ID生成器工具类,此工具类中主要封装:
|
||||
*
|
||||
@ -241,24 +240,26 @@ public class IdUtil {
|
||||
*/
|
||||
return (mpid.toString().hashCode() & 0xffff) % (maxWorkerId + 1);
|
||||
}
|
||||
// ------------------------------------------------------------------- NanoId
|
||||
|
||||
// ------------------------------------------------------------------- NanoId
|
||||
/**
|
||||
* 获取随机NanoId
|
||||
*
|
||||
* @return 随机NanoId
|
||||
* @since 5.7.5
|
||||
*/
|
||||
public static String randomNanoId() {
|
||||
public static String nanoId() {
|
||||
return NanoId.randomNanoId();
|
||||
}
|
||||
|
||||
/**
|
||||
* Static factory to retrieve a NanoId String.
|
||||
* 获取随机NanoId
|
||||
*
|
||||
* The string is generated using the given random number generator.
|
||||
*
|
||||
* @param size The number of symbols in the NanoId String.
|
||||
* @return A randomly generated NanoId String.
|
||||
* @param size ID中的字符数量
|
||||
* @return 随机NanoId
|
||||
* @since 5.7.5
|
||||
*/
|
||||
public static String randomNanoId(final int size){
|
||||
return NanoId.randomNanoId(DEFAULT_NUMBER_GENERATOR, DEFAULT_ALPHABET, size);
|
||||
public static String nanoId(int size){
|
||||
return NanoId.randomNanoId(size);
|
||||
}
|
||||
}
|
||||
|
@ -1,12 +1,9 @@
|
||||
package cn.hutool.core.lang;
|
||||
|
||||
import org.hamcrest.MatcherAssert;
|
||||
import org.hamcrest.Matchers;
|
||||
import cn.hutool.core.lang.id.NanoId;
|
||||
import org.junit.Assert;
|
||||
import org.junit.Test;
|
||||
import org.junit.experimental.results.ResultMatchers;
|
||||
|
||||
import java.lang.reflect.Constructor;
|
||||
import java.lang.reflect.Modifier;
|
||||
import java.security.SecureRandom;
|
||||
import java.util.HashMap;
|
||||
import java.util.HashSet;
|
||||
@ -20,182 +17,154 @@ import static org.junit.Assert.assertTrue;
|
||||
import static org.junit.Assert.fail;
|
||||
|
||||
/**
|
||||
* Tests for NanoIdUtils.
|
||||
* Tests for NanoId.
|
||||
*
|
||||
* @author David Klebanoff
|
||||
* @author David Klebanoff, Looly
|
||||
* @see NanoId
|
||||
*/
|
||||
public class NanoIdTest {
|
||||
|
||||
@Test
|
||||
public void NanoId_VerifyClassIsFinal_Verified() {
|
||||
if ((NanoId.class.getModifiers() & Modifier.FINAL) != Modifier.FINAL) {
|
||||
fail("The class is not final");
|
||||
}
|
||||
}
|
||||
@Test
|
||||
public void nanoIdVerify100KRandomNanoIdsAreUniqueVerifiedTest() {
|
||||
|
||||
@Test
|
||||
public void NanoId_VerifyConstructorsArePrivate_Verified() {
|
||||
for (final Constructor<?> constructor : NanoId.class.getConstructors()) {
|
||||
if ((constructor.getModifiers() & Modifier.PRIVATE) != Modifier.PRIVATE) {
|
||||
fail("The class has a non-private constructor.");
|
||||
}
|
||||
}
|
||||
}
|
||||
//It's not much, but it's a good sanity check I guess.
|
||||
final int idCount = 100000;
|
||||
final Set<String> ids = new HashSet<>(idCount);
|
||||
|
||||
@Test
|
||||
public void NanoId_Verify100KRandomNanoIdsAreUnique_Verified() {
|
||||
for (int i = 0; i < idCount; i++) {
|
||||
final String id = NanoId.randomNanoId();
|
||||
if (ids.contains(id) == false) {
|
||||
ids.add(id);
|
||||
} else {
|
||||
fail("Non-unique ID generated: " + id);
|
||||
}
|
||||
}
|
||||
|
||||
//It's not much, but it's a good sanity check I guess.
|
||||
final int idCount = 100000;
|
||||
final Set<String> ids = new HashSet<>(idCount);
|
||||
}
|
||||
|
||||
for (int i = 0; i < idCount; i++) {
|
||||
final String id = NanoId.randomNanoId();
|
||||
if (ids.contains(id) == false) {
|
||||
ids.add(id);
|
||||
} else {
|
||||
fail("Non-unique ID generated: " + id);
|
||||
}
|
||||
}
|
||||
@Test
|
||||
public void nanoIdSeededRandomSuccessTest() {
|
||||
|
||||
}
|
||||
//With a seed provided, we can know which IDs to expect, and subsequently verify that the
|
||||
// provided random number generator is being used as expected.
|
||||
final Random random = new Random(12345);
|
||||
|
||||
@Test
|
||||
public void NanoId_SeededRandom_Success() {
|
||||
final char[] alphabet =
|
||||
("_-0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ").toCharArray();
|
||||
|
||||
//With a seed provided, we can know which IDs to expect, and subsequently verify that the
|
||||
// provided random number generator is being used as expected.
|
||||
final Random random = new Random(12345);
|
||||
final int size = 21;
|
||||
|
||||
final char[] alphabet =
|
||||
("_-0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ").toCharArray();
|
||||
final String[] expectedIds = new String[]{"kutqLNv1wDmIS56EcT3j7", "U497UttnWzKWWRPMHpLD7",
|
||||
"7nj2dWW1gjKLtgfzeI8eC", "I6BXYvyjszq6xV7L9k2A9", "uIolcQEyyQIcn3iM6Odoa"};
|
||||
|
||||
final int size = 21;
|
||||
for (final String expectedId : expectedIds) {
|
||||
final String generatedId = NanoId.randomNanoId(random, alphabet, size);
|
||||
assertEquals(expectedId, generatedId);
|
||||
}
|
||||
|
||||
final String[] expectedIds = new String[] {"kutqLNv1wDmIS56EcT3j7", "U497UttnWzKWWRPMHpLD7",
|
||||
"7nj2dWW1gjKLtgfzeI8eC", "I6BXYvyjszq6xV7L9k2A9", "uIolcQEyyQIcn3iM6Odoa" };
|
||||
}
|
||||
|
||||
for (final String expectedId : expectedIds) {
|
||||
final String generatedId = NanoId.randomNanoId(random, alphabet, size);
|
||||
assertEquals(expectedId, generatedId);
|
||||
}
|
||||
@Test
|
||||
public void nanoIdVariousAlphabetsSuccessTest() {
|
||||
|
||||
}
|
||||
//Test ID generation with various alphabets consisting of 1 to 255 unique symbols.
|
||||
for (int symbols = 1; symbols <= 255; symbols++) {
|
||||
|
||||
@Test
|
||||
public void NanoId_VariousAlphabets_Success() {
|
||||
final char[] alphabet = new char[symbols];
|
||||
for (int i = 0; i < symbols; i++) {
|
||||
alphabet[i] = (char) i;
|
||||
}
|
||||
|
||||
//Test ID generation with various alphabets consisting of 1 to 255 unique symbols.
|
||||
for (int symbols = 1; symbols <= 255; symbols++) {
|
||||
final String id = NanoId
|
||||
.randomNanoId(null, alphabet, NanoId.DEFAULT_SIZE);
|
||||
|
||||
final char[] alphabet = new char[symbols];
|
||||
for (int i = 0; i < symbols; i++) {
|
||||
alphabet[i] = (char) i;
|
||||
}
|
||||
//Create a regex pattern that only matches to the characters in the alphabet
|
||||
final StringBuilder patternBuilder = new StringBuilder();
|
||||
patternBuilder.append("^[");
|
||||
for (final char character : alphabet) {
|
||||
patternBuilder.append(Pattern.quote(String.valueOf(character)));
|
||||
}
|
||||
patternBuilder.append("]+$");
|
||||
|
||||
final String id = NanoId
|
||||
.randomNanoId(NanoId.DEFAULT_NUMBER_GENERATOR, alphabet,
|
||||
NanoId.DEFAULT_SIZE);
|
||||
assertTrue(id.matches(patternBuilder.toString()));
|
||||
}
|
||||
|
||||
//Create a regex pattern that only matches to the characters in the alphabet
|
||||
final StringBuilder patternBuilder = new StringBuilder();
|
||||
patternBuilder.append("^[");
|
||||
for (final char character : alphabet) {
|
||||
patternBuilder.append(Pattern.quote(String.valueOf(character)));
|
||||
}
|
||||
patternBuilder.append("]+$");
|
||||
}
|
||||
|
||||
assertTrue(id.matches(patternBuilder.toString()));
|
||||
}
|
||||
@Test
|
||||
public void nanoIdVariousSizesSuccessTest() {
|
||||
|
||||
}
|
||||
//Test ID generation with all sizes between 1 and 1,000.
|
||||
for (int size = 1; size <= 1000; size++) {
|
||||
|
||||
@Test
|
||||
public void NanoId_VariousSizes_Success() {
|
||||
final String id = NanoId.randomNanoId(size);
|
||||
|
||||
//Test ID generation with all sizes between 1 and 1,000.
|
||||
for (int size = 1; size <= 1000; size++) {
|
||||
assertEquals(size, id.length());
|
||||
}
|
||||
|
||||
final String id = NanoId.randomNanoId(NanoId.DEFAULT_NUMBER_GENERATOR,
|
||||
NanoId.DEFAULT_ALPHABET, size);
|
||||
}
|
||||
|
||||
assertEquals(size, id.length());
|
||||
}
|
||||
@Test
|
||||
public void nanoIdWellDistributedSuccess() {
|
||||
|
||||
}
|
||||
//Test if symbols in the generated IDs are well distributed.
|
||||
|
||||
@Test
|
||||
public void NanoId_WellDistributed_Success() {
|
||||
final int idCount = 100000;
|
||||
final int idSize = 20;
|
||||
final char[] alphabet = "abcdefghijklmnopqrstuvwxyz".toCharArray();
|
||||
|
||||
//Test if symbols in the generated IDs are well distributed.
|
||||
final Map<String, Long> charCounts = new HashMap<>();
|
||||
|
||||
final int idCount = 100000;
|
||||
final int idSize = 20;
|
||||
final char[] alphabet = "abcdefghijklmnopqrstuvwxyz".toCharArray();
|
||||
for (int i = 0; i < idCount; i++) {
|
||||
|
||||
final Map<String, Long> charCounts = new HashMap<>();
|
||||
final String id = NanoId
|
||||
.randomNanoId(null, alphabet, idSize);
|
||||
|
||||
for (int i = 0; i < idCount; i++) {
|
||||
for (int j = 0; j < id.length(); j++) {
|
||||
final String value = String.valueOf(id.charAt(j));
|
||||
|
||||
final String id = NanoId
|
||||
.randomNanoId(NanoId.DEFAULT_NUMBER_GENERATOR, alphabet, idSize);
|
||||
final Long charCount = charCounts.get(value);
|
||||
if (charCount == null) {
|
||||
charCounts.put(value, 1L);
|
||||
} else {
|
||||
charCounts.put(value, charCount + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (int j = 0; j < id.length(); j++) {
|
||||
final String value = String.valueOf(id.charAt(j));
|
||||
//Verify the distribution of characters is pretty even
|
||||
for (final Long charCount : charCounts.values()) {
|
||||
final double distribution = (charCount * alphabet.length / (double) (idCount * idSize));
|
||||
Assert.assertTrue(distribution >= 0.95 && distribution <= 1.05);
|
||||
}
|
||||
|
||||
final Long charCount = charCounts.get(value);
|
||||
if (charCount == null) {
|
||||
charCounts.put(value, 1L);
|
||||
} else {
|
||||
charCounts.put(value, charCount + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//Verify the distribution of characters is pretty even
|
||||
for (final Long charCount : charCounts.values()) {
|
||||
final double distribution = (charCount * alphabet.length / (double) (idCount * idSize));
|
||||
MatcherAssert.assertThat(distribution, Matchers.closeTo(1.0, 0.05));
|
||||
}
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoIdEmptyAlphabetExceptionThrownTest() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[]{}, 10);
|
||||
}
|
||||
|
||||
}
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId256AlphabetExceptionThrownTest() {
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_NullRandom_ExceptionThrown() {
|
||||
NanoId.randomNanoId(null, new char[] {'a', 'b', 'c'}, 10);
|
||||
}
|
||||
//The alphabet is composed of 256 unique characters
|
||||
final char[] largeAlphabet = new char[256];
|
||||
for (int i = 0; i < 256; i++) {
|
||||
largeAlphabet[i] = (char) i;
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_NullAlphabet_ExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), null, 10);
|
||||
}
|
||||
NanoId.randomNanoId(new SecureRandom(), largeAlphabet, 20);
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_EmptyAlphabet_ExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[] {}, 10);
|
||||
}
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_256Alphabet_ExceptionThrown() {
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoIdNegativeSizeExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[]{'a', 'b', 'c'}, -10);
|
||||
}
|
||||
|
||||
//The alphabet is composed of 256 unique characters
|
||||
final char[] largeAlphabet = new char[256];
|
||||
for (int i = 0; i < 256; i++) {
|
||||
largeAlphabet[i] = (char) i;
|
||||
}
|
||||
|
||||
NanoId.randomNanoId(new SecureRandom(), largeAlphabet, 20);
|
||||
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_NegativeSize_ExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[] {'a', 'b', 'c'}, -10);
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoId_ZeroSize_ExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[] {'a', 'b', 'c'}, 0);
|
||||
}
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void randomNanoIdZeroSizeExceptionThrown() {
|
||||
NanoId.randomNanoId(new SecureRandom(), new char[]{'a', 'b', 'c'}, 0);
|
||||
}
|
||||
}
|
||||
|
@ -1,9 +1,11 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
||||
|
||||
<parent>
|
||||
<groupId>cn.hutool</groupId>
|
||||
<artifactId>hutool-parent</artifactId>
|
||||
@ -13,7 +15,7 @@
|
||||
<artifactId>hutool-cron</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool 定时任务</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,9 +1,11 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
||||
|
||||
<parent>
|
||||
<groupId>cn.hutool</groupId>
|
||||
<artifactId>hutool-parent</artifactId>
|
||||
@ -13,7 +15,7 @@
|
||||
<artifactId>hutool-dfa</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool 基于DFA的关键词查找</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,5 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,6 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,5 +1,6 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
@ -15,7 +15,7 @@
|
||||
<artifactId>hutool-socket</artifactId>
|
||||
<name>${project.artifactId}</name>
|
||||
<description>Hutool套接字,包括BIO、NIO、AIO封装</description>
|
||||
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>cn.hutool</groupId>
|
||||
|
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>jar</packaging>
|
||||
|
10
pom.xml
10
pom.xml
@ -1,7 +1,7 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<packaging>pom</packaging>
|
||||
@ -55,12 +55,6 @@
|
||||
<version>${junit.version}</version>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.hamcrest</groupId>
|
||||
<artifactId>java-hamcrest</artifactId>
|
||||
<version>2.0.0.0</version>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.projectlombok</groupId>
|
||||
<artifactId>lombok</artifactId>
|
||||
|
Loading…
Reference in New Issue
Block a user