一套优秀的随机字符串生成机制不仅需要确保全局唯一性,还需要考虑到性能等问题。翻看了之前写过的一篇生成三位整数随机数的博文《Java 生成三位随机数》,发现她满足了当时的需求,现在提供一个基于她的plus版本,提高随机数的应用范围和复杂度。本篇文章带大家探究一下如何用字母和数字生成随机字符串。

前言

用字母和数字随机生成不重复的字符串需要满足如下条件:

  • 唯一性:确保每个字符串都是系统中独一无二的,使得可以根据此字符串进行反向推导。
  • 随机性:满足难于从字符串上推断出生成机制的要求。
  • 高效性:算法简单,时间复杂度低,或者不过度耗费系统资源。
  • 简洁性:用户可以方便的识别。

例如,我们平时看到的邀请码、短信验证码、快递代收点推送的取件码或者图形验证码,就是一组随机字符串,由此可见,随机字符串的应用非常广,老铁们需要了解一下她的生成机制。这些随机字符串的长度通常在6位左右,就是为了满足简洁性。当然,对人人皆知的java.util.UUID UUID.randomUUID().toString()就不再介绍了。

生成随机字符串

这里提供一个随机生成字符串的工具类,里面包括五种生成机制,源码如下。当然,为了达到不重复的目的,需要在数据库中创建一张专门维护已使用随机字符串的表tabA,在生成随机字符串后,根据此字符串到tabA中查重,如果存在,则继续生成新的字符串,直到拿到不重复的字符串为止。如果重复次数比较多,则可以适度增加随机字符串的长度,例如方法generateByRandom(final int length),字符串长度length可以由业务实际需求确定。

import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Random; /**
* 用字母和数字生成不重复的随机字符串
*
* @author Wiener
* @date 2020/11/17
*/
public class RandomUtil {
/**
* 字符源,可以剔除O、L、0和1,避免0和1与O和L混淆,这里没有剔除<br/>
* 可以根据需要加入小写英文字母和特殊字符等
*/
private static final String[] GENERATE_SOURCE = new String[]{"0", "1", "2", "3", "4", "5", "6", "7",
"8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J",
"K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V",
"W", "X", "Y", "Z"};
private static final int STR_LEN = GENERATE_SOURCE.length; /**
* 使用 Collections.shuffle 生成六位随机字符串
*
* @return
*/
private static String generateByShuffle() {
List<String> list = Arrays.asList(GENERATE_SOURCE);
//打乱元素排序,增加反推难度
Collections.shuffle(list);
StringBuilder randomStr = new StringBuilder();
for (int i = 0; i < STR_LEN; i++) {
randomStr.append(list.get(i));
}
//更改下面两个数字可以取到不同位数的随机数哦
return randomStr.substring(4, 10);
} public static void main(String[] args) {
for (int i = 0; i < 10; i++) {
System.out.println(generateByRandom(6));
}
System.out.println(generateByShuffle() + "---" + STR_LEN);
} /**
* 生成数字和字母组合,字母区分大小写
*
* @param length 随机字符串的长度
* @return
*/
public static String generateByRandom(final int length) {
StringBuilder randomSb = new StringBuilder(length);
Random random = new Random();
for (int i = 0; i < length; i++) {
// 输出字母还是数字
String charOrNum = random.nextInt(2) % 2 == 0 ? "char" : "num";
// 字符串
if ("char".equals(charOrNum)) {
// 判断字母大小写
int choice = random.nextInt(2) % 2 == 0 ? 65 : 97;
randomSb = randomSb.append((char) (choice + random.nextInt(26)));
} else {
randomSb = randomSb.append(random.nextInt(10));
}
}
return randomSb.toString();
}
/**
* 生成随机字符串,generateByRandom的简化版
* @param count 随机字符串的长度
* @param source 源字符集
* @return
*/
public static String doGenerate(int count, String[] source) {
StringBuilder sb = new StringBuilder(count);
int sourceLen = source.length;
for (int i = 0; i < count; i++) {
final int index = new Random().nextInt(sourceLen);
sb.append(source[index]);
}
return sb.toString();
}
/**
* @param begin 自增长序列
* @return
*/
public static String generateByOrder(final int begin) {
List<String> src = Arrays.asList(GENERATE_SOURCE);
Collections.shuffle(src);
StringBuilder randomSb = new StringBuilder(6);
int i4 = (begin) % 36;
int i3 = (begin / (36)) % 36;
int i2 = (begin / (36 * 36)) % 36;
int i1 = (begin / (36 * 36 * 36)) % 36;
int i0 = (begin / (36 * 36 * 36 * 36)) % 36;
int i = (begin / (36 * 36 * 36 * 36 * 36)) % 36;
randomSb = randomSb.append(src.get(i0)).append(src.get(i1))
.append(src.get(i2)).append(src.get(i3))
.append(src.get(i4)).append(src.get(i));
return randomSb.toString();
} }

第一种生成机制generateByShuffle()基于Collections.shuffle 生成六位随机字符串。generateByRandom(int length)借助Random()函数生成区分字母大小写的随机字符串,字符串长度可以根据入参自定义,提供了更灵活的生成机制。doGenerate(int count, String[] source) 是generateByRandom(int length)的简化版。第四种生成机制generateByOrder(int begin)需要在使用时维护一个自增长序列,保证入参begin是自增长的,举个例子:

    public static void main(String[] args) {
for (int i = 0; i < 100; i = i+10) {
System.out.println(generateByOrder(i));
}
}

第五种生成机制是直接使用randomAlphanumericc(final int count)函数,她位于org.apache.commons.lang3包下的RandomStringUtils类中,可以随机生成指定长度为count的字符串。

代码如下:

    /**
* @param count the length of random string to create
* @return
*/
public static String randomAlphanumeric(int count) {
return RandomStringUtils.randomAlphanumeric(count);
}

此方法需要导入如下依赖:

<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
<version>3.7</version>
</dependency>

RandomStringUtils.randomAlphanumeric(count)生成一个指定长度为count的随机字符串,内容为大小写字母和0~9的数字。另外,RandomStringUtils类中的函数randomAscii(final int count)可以生成从ASCII 32到126组成的随机字符串,长度为count。

结束语

本文介绍了四种用26个字母和10个数字,随机生成一个不重复6位字符串的策略。当业务不断发展,如果36^6个字符串依旧无法满足业务需求,则可采用以下方式进行扩充:

  • 扩充邀请码位数,比如变为8位或者更多位。
  • 扩充字符源,比如加入小写字母,加减号等特殊字符。第二种和第四种生成机制已经使用了小写字母。

欢迎点赞阅读,一同学习交流;若有疑问,请在文章下方留言!

Reference

https://www.cnblogs.com/itbac/p/11148159.html

Java 生成随机字符串的六种方法的更多相关文章

  1. 生成随机字符串(UUID方法)

    这是另一种用UUID生成随机字符串的方法. public class RandomGenerator{ private int length; public void setLength(int le ...

  2. JS生成随机字符串的多种方法

    这篇文章主要介绍了JS生成随机字符串的方法,需要的朋友可以参考下 下面的一段代码,整理电脑时,记录备查. <script language="javascript"> ...

  3. java生成随机字符串

    学习java comparable特性时候,定义如下Student类,需要需要随机添加学生姓名以及学号和成绩,这是java如何随机生成名字,根据我的查询,我找到目前java库支持两种方法. 1. or ...

  4. java 生成随机字符串

    1.生成之指定位数的随机字符串 /** * 随机基数 */ private static char[] charset = {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h ...

  5. java生成随机字符串uuid

    GUID是一个128位长的数字,一般用16进制表示.算法的核心思想是结合机器的网卡.当地时间.一个随即数来生成GUID.从理论上讲,如果一台机器每秒产生10000000个GUID,则可以保证(概率意义 ...

  6. java使用apache-commons-lang3生成随机字符串(可自定义规则)

    在日常开发中,我们经常会遇到生成随机字符串的需求.可能是大小写字母+数字,也可能是其他各种字符.作为一个常用功能,我们完全没必要自己实现,有很多优质的类库已经做的很完善了.本文介绍的就是apache- ...

  7. (C#)生成指定长度的随机字符串的通用方法

    .NET(C#)生成指定长度的随机字符串的通用方法,此方法可以指定字符串的长度,是否包含数字,是否包含符号,是否包含小写字母,是否包含大写字母等, 源码: #region 生成指定长度的随机字符串 / ...

  8. PHP 生成随机字符串与唯一字符串

    说明:生成随机字符串用到的方法有 mt_rand() 生成唯一字符串用到的方法有 md5(),uniqid(),microtime() 代码: <?php /* * 生成随机字符串 * @par ...

  9. PHP生成随机字符串包括大小写字母

    PHP生成随机字符串包括大小写字母,这里介绍两种方法: 第一种:利用字符串函数操作 <?php /** *@blog <www.phpddt.com> */ function cre ...

  10. 【转】Java生成对应字符串的MD5密码模块

    原文网址:http://www.cnblogs.com/xudong-bupt/archive/2013/05/10/3070899.html (1)一般使用的数据库中都会保存用户名和密码,其中密码不 ...

随机推荐

  1. Visual Studio 好用的主题+字体推荐!!!

    Vs2022主题+字体 Visual Studio(VS)是一款功能强大的集成开发环境(IDE),可以用于开发各种类型的应用程序,包括桌面应用.Web应用.移动应用等.它提供了许多主题设置和字体选项, ...

  2. C# 生成缩略图方法

    private static string CreateThumbnail(string filepath, int tWidth, int tHeight) { if (string.IsNullO ...

  3. 震惊!AI编程正在淘汰这5类人,你在其中吗?

    大家好,我是狂师. 今天在知乎上看到一个关于讨论:"人工智能大爆发,AI编程工具对程序员到底是颠覆还是辅助?'"问题,觉得蛮有意思.的确,AI编程的出现,引发了人们对于程序员职业未 ...

  4. 第二届獬豸杯wp

    第二届獬豸杯wp 容器密码:}2N|n_yxdt!G/Ru}|_zdn$@?6@CD8E 计算机和手机部分已经在第二届 獬豸杯-复现 - 萧瑟迪亲传大弟子 - 博客园这里发过了,服务器部分自己又写了一 ...

  5. 【数值方法-Python实现】Crout分解+追赶法实现

    涉及Crout分解.追赶法的线性方程组求解方法的Python实现. Codes def CroutLU(A:np.ndarray)->Tuple[np.ndarray,np.ndarray]: ...

  6. 记录composer 安装 yii2项目

    先带上一个痛苦面具 前段时间换成mac系统,自己以前的yii2项目老是安装不上,因为暂时用不上就没去管,现在想用了,折腾了半天才安装好.下面我记录下坑 国内记得换镜像(我换了系统后,应该是忘记了) c ...

  7. 22. Generate Parentheses--求n对括号组成可以组成的全部有效括号序列

    描述: Given n pairs of parentheses, write a function to generate all combinations of well-formed paren ...

  8. CentOS 版本选择DVD、Everything、LiveCD、Minimal、NetInstall

    CentOS 7.X,主要是下载的时候有很多版本供选择,如何选择? DVD版:这个是常用版本,就是普通安装版了,推荐大家安装.里面包含大量的常用软件,大部分情况下安装时无需再在线下载,体积为4G.Ev ...

  9. linux下的nginx重启命令常见以下3种:

    systemctl restart nginx service nginx restart /usr/sbin/nginx -s reload

  10. NextJS CVE-2025-29927 安全漏洞

    NextJS CVE-2025-29927 安全漏洞 CVE-2025-29927 是一个存在于 Next.js 框架中的关键安全漏洞.该漏洞允许攻击者通过伪造或篡改 x-middleware-sub ...