com.neko233.toolchain.common.base.RandomUtils233 Maven / Gradle / Ivy
package com.neko233.toolchain.common.base;
import com.neko233.toolchain.common.exception.UtilException;
import com.neko233.toolchain.common.random.WeightObject;
import com.neko233.toolchain.common.random.WeightRandom;
import org.joda.time.DateTime;
import org.joda.time.LocalDateTime;
import java.awt.*;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.List;
import java.util.*;
import java.util.concurrent.ThreadLocalRandom;
import java.util.stream.Collectors;
public class RandomUtils233 {
/**
* 用于随机选的数字
*/
public static final String BASE_NUMBER = "0123456789";
/**
* 用于随机选的字符
*/
public static final String BASE_CHAR = "abcdefghijklmnopqrstuvwxyz";
/**
* 用于随机选的字符和数字
*/
public static final String BASE_CHAR_NUMBER = BASE_CHAR + BASE_NUMBER;
/**
* 获取随机数生成器对象
* ThreadLocalRandom是JDK 7之后提供并发产生随机数,能够解决多个线程发生的竞争争夺。
*
*
* 注意:此方法返回的{@link ThreadLocalRandom}不可以在多线程环境下共享对象,否则有重复随机数问题。
* 见:https://www.jianshu.com/p/89dfe990295c
*
*
* @return {@link ThreadLocalRandom}
*/
public static ThreadLocalRandom getRandom() {
return ThreadLocalRandom.current();
}
/**
* 创建{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
*
* @param seed 自定义随机种子
* @return {@link SecureRandom}
*/
public static SecureRandom createSecureRandom(byte[] seed) {
return (null == seed) ? new SecureRandom() : new SecureRandom(seed);
}
/**
* 获取SHA1PRNG的{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
* 注意:此方法获取的是伪随机序列发生器PRNG(pseudo-random number generator)
*
*
* 相关说明见:https://stackoverflow.com/questions/137212/how-to-solve-slow-java-securerandom
*
* @return {@link SecureRandom}
*/
public static SecureRandom getSecureRandom() {
return getSecureRandom(null);
}
/**
* 获取SHA1PRNG的{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
* 注意:此方法获取的是伪随机序列发生器PRNG(pseudo-random number generator)
*
*
* 相关说明见:https://stackoverflow.com/questions/137212/how-to-solve-slow-java-securerandom
*
* @param seed 随机数种子
* @return {@link SecureRandom}
* @see #createSecureRandom(byte[])
*/
public static SecureRandom getSecureRandom(byte[] seed) {
return createSecureRandom(seed);
}
/**
* 获取SHA1PRNG的{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
* 注意:此方法获取的是伪随机序列发生器PRNG(pseudo-random number generator),在Linux下噪声生成时可能造成较长时间停顿。
* see: http://ifeve.com/jvm-random-and-entropy-source/
*
*
* 相关说明见:https://stackoverflow.com/questions/137212/how-to-solve-slow-java-securerandom
*
* @param seed 随机数种子
* @return {@link SecureRandom}
*/
public static SecureRandom getSHA1PRNGRandom(byte[] seed) {
SecureRandom random;
try {
random = SecureRandom.getInstance("SHA1PRNG");
} catch (NoSuchAlgorithmException e) {
throw new UtilException(e);
}
if (null != seed) {
random.setSeed(seed);
}
return random;
}
/**
* 获取algorithms/providers中提供的强安全随机生成器
* 注意:此方法可能造成阻塞或性能问题
*
* @return {@link SecureRandom}
*/
public static SecureRandom getSecureRandomStrong() {
try {
return SecureRandom.getInstanceStrong();
} catch (NoSuchAlgorithmException e) {
throw new UtilException(e);
}
}
/**
* 获取随机数产生器
*
* @param isSecure 是否为强随机数生成器 (RNG)
* @return {@link Random}
* @see #getSecureRandom()
* @see #getRandom()
*/
public static Random getRandom(boolean isSecure) {
return isSecure ? getSecureRandom() : getRandom();
}
/**
* 获得随机Boolean值
*
* @return true or false
*/
public static boolean randomBoolean() {
return 0 == randomInt(2);
}
/**
* 随机汉字('\u4E00'-'\u9FFF')
*
* @return 随机的汉字字符
*/
public static char randomChinese() {
return (char) randomInt('\u4E00', '\u9FFF');
}
/**
* 获得指定范围内的随机数
*
* @param min 最小数(包含)
* @param max 最大数(不包含)
* @return 随机数
*/
public static int randomInt(int min, int max) {
return getRandom().nextInt(min, max);
}
/**
* 获得随机数int值
*
* @return 随机数
* @see Random#nextInt()
*/
public static int randomInt() {
return getRandom().nextInt();
}
/**
* 获得指定范围内的随机数 [0,limit)
*
* @param limit 限制随机数的范围,不包括这个数
* @return 随机数
* @see Random#nextInt(int)
*/
public static int randomInt(int limit) {
return getRandom().nextInt(limit);
}
/**
* 获得指定范围内的随机数[min, max)
*
* @param min 最小数(包含)
* @param max 最大数(不包含)
* @return 随机数
* @see ThreadLocalRandom#nextLong(long, long)
*/
public static long randomLong(long min, long max) {
return getRandom().nextLong(min, max);
}
/**
* 获得随机数
*
* @return 随机数
* @see ThreadLocalRandom#nextLong()
*/
public static long randomLong() {
return getRandom().nextLong();
}
/**
* 获得指定范围内的随机数 [0,limit)
*
* @param limit 限制随机数的范围,不包括这个数
* @return 随机数
* @see ThreadLocalRandom#nextLong(long)
*/
public static long randomLong(long limit) {
return getRandom().nextLong(limit);
}
/**
* 获得指定范围内的随机数
*
* @param min 最小数(包含)
* @param max 最大数(不包含)
* @return 随机数
* @see ThreadLocalRandom#nextDouble(double, double)
*/
public static double randomDouble(double min, double max) {
return getRandom().nextDouble(min, max);
}
/**
* 获得指定范围内的随机数
*
* @param min 最小数(包含)
* @param max 最大数(不包含)
* @param scale 保留小数位数
* @param roundingMode 保留小数的模式 {@link RoundingMode}
* @return 随机数
*/
public static double randomDouble(double min, double max, int scale, RoundingMode roundingMode) {
return NumberUtils233.round(randomDouble(min, max), scale, roundingMode).doubleValue();
}
/**
* 获得随机数[0, 1)
*
* @return 随机数
* @see ThreadLocalRandom#nextDouble()
*/
public static double randomDouble() {
return getRandom().nextDouble();
}
/**
* 获得指定范围内的随机数
*
* @param scale 保留小数位数
* @param roundingMode 保留小数的模式 {@link RoundingMode}
* @return 随机数
*/
public static double randomDouble(int scale, RoundingMode roundingMode) {
return NumberUtils233.round(randomDouble(), scale, roundingMode).doubleValue();
}
/**
* 获得指定范围内的随机数 [0,limit)
*
* @param limit 限制随机数的范围,不包括这个数
* @return 随机数
* @see ThreadLocalRandom#nextDouble(double)
*/
public static double randomDouble(double limit) {
return getRandom().nextDouble(limit);
}
/**
* 获得指定范围内的随机数
*
* @param limit 限制随机数的范围,不包括这个数
* @param scale 保留小数位数
* @param roundingMode 保留小数的模式 {@link RoundingMode}
* @return 随机数
*/
public static double randomDouble(double limit, int scale, RoundingMode roundingMode) {
return NumberUtils233.round(randomDouble(limit), scale, roundingMode).doubleValue();
}
/**
* 获得指定范围内的随机数[0, 1)
*
* @return 随机数
*/
public static BigDecimal randomBigDecimal() {
return NumberUtils233.toBigDecimal(getRandom().nextDouble());
}
/**
* 获得指定范围内的随机数 [0,limit)
*
* @param limit 最大数(不包含)
* @return 随机数
*/
public static BigDecimal randomBigDecimal(BigDecimal limit) {
return NumberUtils233.toBigDecimal(getRandom().nextDouble(limit.doubleValue()));
}
/**
* 获得指定范围内的随机数
*
* @param min 最小数(包含)
* @param max 最大数(不包含)
* @return 随机数
*/
public static BigDecimal randomBigDecimal(BigDecimal min, BigDecimal max) {
return NumberUtils233.toBigDecimal(getRandom().nextDouble(min.doubleValue(), max.doubleValue()));
}
/**
* 随机bytes
*
* @param length 长度
* @return bytes
*/
public static byte[] randomBytes(int length) {
byte[] bytes = new byte[length];
getRandom().nextBytes(bytes);
return bytes;
}
/**
* 随机获得列表中的元素
*
* @param 元素类型
* @param list 列表
* @return 随机元素
*/
public static T randomEle(List list) {
return randomEle(list, list.size());
}
/**
* 随机获得列表中的元素
*
* @param 元素类型
* @param list 列表
* @param limit 限制列表的前N项
* @return 随机元素
*/
public static T randomEle(List list, int limit) {
if (list.size() < limit) {
limit = list.size();
}
return list.get(randomInt(limit));
}
/**
* 随机获得数组中的元素
*
* @param 元素类型
* @param array 列表
* @return 随机元素
*/
public static T randomEle(T[] array) {
return randomEle(array, array.length);
}
/**
* 随机获得数组中的元素
*
* @param 元素类型
* @param array 列表
* @param limit 限制列表的前N项
* @return 随机元素
*/
public static T randomEle(T[] array, int limit) {
if (array.length < limit) {
limit = array.length;
}
return array[randomInt(limit)];
}
/**
* 随机获得列表中的一定量元素
*
* @param 元素类型
* @param list 列表
* @param count 随机取出的个数
* @return 随机元素
*/
public static List randomObject(List list, int count) {
final List result = new ArrayList<>(count);
int limit = list.size();
while (result.size() < count) {
result.add(randomEle(list, limit));
}
return result;
}
/**
* 随机获得列表中的一定量的元素,返回List
* 此方法与{@link #randomObject(List, int)} 不同点在于,不会获取重复位置的元素
*
* @param source 列表
* @param needCount 随机取出的个数
* @param 元素类型
* @return 随机列表
*/
public static List randomObjectList(List source, int needCount) {
if (needCount >= source.size()) {
return new ArrayList<>(source);
}
List result = new ArrayList<>();
final int[] randomCountIndexList = randomInts(needCount);
for (int e : randomCountIndexList) {
result.add(source.get(e));
}
return result;
}
/**
* 随机获得列表中的一定量的不重复元素,返回Set
*
* @param 元素类型
* @param collection 列表
* @param count 随机取出的个数
* @return 随机元素
* @throws IllegalArgumentException 需要的长度大于给定集合非重复总数
*/
public static Set randomObjectSet(Collection collection, int count) {
final List source = collection.stream().distinct().collect(Collectors.toList());
if (count > source.size()) {
throw new IllegalArgumentException("Count is larger than collection distinct size !");
}
final Set result = new LinkedHashSet<>(count);
int limit = source.size();
while (result.size() < count) {
result.add(randomEle(source, limit));
}
return result;
}
/**
* 创建指定长度的随机索引
*
* @param length 长度
* @return 随机索引
*/
public static int[] randomInts(int length) {
final int[] range = ArrayUtils233.range(length);
for (int i = 0; i < length; i++) {
int random = randomInt(i, length);
ArrayUtils233.swap(range, i, random);
}
return range;
}
/**
* 获得一个随机的字符串(只包含数字和字符)
*
* @param length 字符串的长度
* @return 随机字符串
*/
public static String randomString(int length) {
return randomString(BASE_CHAR_NUMBER, length);
}
/**
* 获得一个随机的字符串(只包含数字和大写字符)
*
* @param length 字符串的长度
* @return 随机字符串
*/
public static String randomStringUpper(int length) {
return randomString(BASE_CHAR_NUMBER, length).toUpperCase();
}
/**
* 获得一个随机的字符串(只包含数字和小写字母) 并排除指定字符串
*
* @param length 字符串的长度
* @param elemData 要排除的字符串,如:去重容易混淆的字符串,oO0、lL1、q9Q、pP,不区分大小写
* @return 随机字符串
*/
public static String randomStringWithoutStr(int length, String elemData) {
String baseStr = BASE_CHAR_NUMBER;
baseStr = StringUtils233.removeAll(baseStr, elemData.toLowerCase().toCharArray());
return randomString(baseStr, length);
}
/**
* 获得一个只包含数字的字符串
*
* @param length 字符串的长度
* @return 随机字符串
*/
public static String randomNumbers(int length) {
return randomString(BASE_NUMBER, length);
}
/**
* 获得一个随机的字符串
*
* @param baseString 随机字符选取的样本
* @param length 字符串的长度
* @return 随机字符串
*/
public static String randomString(String baseString, int length) {
if (StringUtils233.isEmpty(baseString)) {
return StringUtils233.EMPTY;
}
if (length < 1) {
length = 1;
}
final StringBuilder sb = new StringBuilder(length);
int baseLength = baseString.length();
for (int i = 0; i < length; i++) {
int number = randomInt(baseLength);
sb.append(baseString.charAt(number));
}
return sb.toString();
}
/**
* 随机数字,数字为0~9单个数字
*
* @return 随机数字字符
*/
public static char randomNumber() {
return randomChar(BASE_NUMBER);
}
/**
* 随机字母或数字,小写
*
* @return 随机字符
*/
public static char randomChar() {
return randomChar(BASE_CHAR_NUMBER);
}
/**
* 随机字符
*
* @param baseString 随机字符选取的样本
* @return 随机字符
*/
public static char randomChar(String baseString) {
return baseString.charAt(randomInt(baseString.length()));
}
/**
* 生成随机颜色
*
* @return 随机颜色
* @deprecated 使用ImgUtil.randomColor()
*/
@Deprecated
public static Color randomColor() {
final Random random = getRandom();
return new Color(random.nextInt(256), random.nextInt(256), random.nextInt(256));
}
/**
* 带有权重的随机生成器
*
* @param 随机对象类型
* @param weightObjects 带有权重的对象列表
* @return {@link WeightRandom}
*/
public static WeightRandom weightRandom(WeightObject[] weightObjects) {
return new WeightRandom<>(weightObjects);
}
/**
* 带有权重的随机生成器
*
* @param 随机对象类型
* @param weightObjs 带有权重的对象列表
* @return {@link WeightRandom}
*/
public static WeightRandom weightRandom(Iterable> weightObjs) {
return new WeightRandom<>(weightObjs);
}
/**
* 以当天为基准,随机产生一个日期. 单位天数
*
* @param min 偏移最小天,可以为负数表示过去的时间(包含)
* @param max 偏移最大天,可以为负数表示过去的时间(不包含)
* @return 随机日期(随机天,其它时间不变)
*/
public static DateTime randomDay(int min, int max) {
int randomInt = randomInt(min, max);
return LocalDateTime.now()
.plusDays(randomInt)
.toDateTime();
}
}