256

在 JavaScript 中生成随机字母数字(大写、小写和数字)字符串以用作可能唯一标识符的最短方法(在合理范围内)是什么?

4

19 回答 19

397

我刚刚发现这是一个非常好的和优雅的解决方案:

Math.random().toString(36).slice(2)

关于此实现的注意事项:

  • 这将产生一个长度介于 0 到 12 个字符之间的字符串,通常为 11 个字符,因为浮点字符串化会删除尾随零。
  • 它不会生成大写字母,只会生成小写字母和数字。
  • 因为随机性来自Math.random(),所以输出可能是可预测的,因此不一定是唯一的。
  • 即使假设是一个理想的实现,输出也最多有 52 位的熵,这意味着在生成大约 70M 的字符串后,您可以预期会出现重复。
于 2012-09-19T20:40:12.513 回答
336

如果您只想允许特定字符,您也可以这样做:

function randomString(length, chars) {
    var result = '';
    for (var i = length; i > 0; --i) result += chars[Math.floor(Math.random() * chars.length)];
    return result;
}
var rString = randomString(32, '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ');

这是一个 jsfiddle 来演示:http: //jsfiddle.net/wSQBx/

另一种方法是使用一个特殊的字符串来告诉函数使用什么类型的字符。你可以这样做:

function randomString(length, chars) {
    var mask = '';
    if (chars.indexOf('a') > -1) mask += 'abcdefghijklmnopqrstuvwxyz';
    if (chars.indexOf('A') > -1) mask += 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
    if (chars.indexOf('#') > -1) mask += '0123456789';
    if (chars.indexOf('!') > -1) mask += '~`!@#$%^&*()_+-={}[]:";\'<>?,./|\\';
    var result = '';
    for (var i = length; i > 0; --i) result += mask[Math.floor(Math.random() * mask.length)];
    return result;
}

console.log(randomString(16, 'aA'));
console.log(randomString(32, '#aA'));
console.log(randomString(64, '#A!'));

小提琴:http: //jsfiddle.net/wSQBx/2/

或者,要使用如下所述的 base36 方法,您可以执行以下操作:

function randomString(length) {
    return Math.round((Math.pow(36, length + 1) - Math.random() * Math.pow(36, length))).toString(36).slice(1);
}
于 2012-05-23T20:13:58.367 回答
37

JAR.JAR.beans 提出的另一种答案变体

(Math.random()*1e32).toString(36)

通过更改乘数1e32,您可以更改随机字符串的长度。

于 2015-06-18T20:49:18.747 回答
37

更新:单行解决方案,随机 20 个字符(字母数字小写):

Array.from(Array(20), () => Math.floor(Math.random() * 36).toString(36)).join('');

或者用 lodash 更短:

_.times(20, () => _.random(35).toString(36)).join('');
于 2017-06-19T04:33:10.920 回答
24

或者以 Jar Jar 的建议为基础,这是我在最近的一个项目中使用的(以克服长度限制):

var randomString = function (len, bits)
{
    bits = bits || 36;
    var outStr = "", newStr;
    while (outStr.length < len)
    {
        newStr = Math.random().toString(bits).slice(2);
        outStr += newStr.slice(0, Math.min(newStr.length, (len - outStr.length)));
    }
    return outStr.toUpperCase();
};

采用:

randomString(12, 16); // 12 hexadecimal characters
randomString(200); // 200 alphanumeric characters
于 2014-02-25T23:51:44.017 回答
21

这个更干净

Math.random().toString(36).substr(2, length)

例子

Math.random().toString(36).substr(2, 5)
于 2016-07-04T16:58:22.353 回答
16

我认为以下是允许给定长度的最简单的解决方案:

Array(myLength).fill(0).map(x => Math.random().toString(36).charAt(2)).join('')

这取决于箭头函数语法。

于 2018-01-04T00:33:08.533 回答
10
function randomString(len) {
    var p = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
    return [...Array(len)].reduce(a=>a+p[~~(Math.random()*p.length)],'');
}

概括:

  1. 创建一个我们想要的大小的数组(因为在 javascript.js 中没有range(len)等价物。
  2. 对于数组中的每个元素:从中选择一个随机字符p并将其添加到字符串中
  3. 返回生成的字符串。

一些解释:

[...Array(len)]

Array(len) 或 new Array(len)创建一个带有未定义指针的数组。单线将更难实现。Spread 语法方便地定义了指针(现在它们指向未定义的对象!)。

.reduce(

在这种情况下,将数组缩减为单个字符串。reduce 功能在大多数语言中都很常见,值得学习。

a=>a+...

我们正在使用箭头函数

a累加器。在这种情况下,它是我们完成后要返回的最终结果字符串(你知道它是一个字符串,因为 reduce 函数的第二个参数,initialValue是一个空字符串:) ''。所以基本上:用 转换数组中的每个元素p[~~(Math.random()*p.length)],将结果附加到a字符串并a在完成后给我。

p[...]

p是我们从中选择的字符串。您可以像索引一样访问字符串中的字符(例如,"abcdefg"[3]给我们"d"

~~(Math.random()*p.length)

Math.random()返回 [0, 1) 之间的浮点数Math.floor(Math.random()*max) 是在 javascript 中获取随机整数的事实标准~是 javascript 中的按位 NOT 运算符。 ~~是一种更短,可以说有时更快,而且绝对更有趣的表达方式Math.floor( 这里有一些信息

于 2017-06-21T14:07:31.770 回答
8

32 个字符:

for(var c = ''; c.length < 32;) c += Math.random().toString(36).substr(2, 1)
于 2014-06-30T19:01:05.353 回答
7

使用 lodash:

function createRandomString(length) {
        var chars = "abcdefghijklmnopqrstufwxyzABCDEFGHIJKLMNOPQRSTUFWXYZ1234567890"
        var pwd = _.sampleSize(chars, length || 12)  // lodash v4: use _.sampleSize
        return pwd.join("")
    }
document.write(createRandomString(8))
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.11/lodash.min.js"></script>

于 2016-04-26T07:54:09.617 回答
6

随机字符:

String.fromCharCode(i); //where is an int

随机整数:

Math.floor(Math.random()*100);

把它们放在一起:

function randomNum(hi){
    return Math.floor(Math.random()*hi);
} 
function randomChar(){
    return String.fromCharCode(randomNum(100));
}
function randomString(length){
   var str = "";
   for(var i = 0; i < length; ++i){
        str += randomChar();
   }
   return str;
}
var RandomString = randomString(32); //32 length string

小提琴:http: //jsfiddle.net/maniator/QZ9J2/

于 2012-05-23T19:56:26.963 回答
4

随机密钥生成器

keyLength 参数是您想要的键的字符长度

function keyGen(keyLength) {
    var i, key = "", characters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";

    var charactersLength = characters.length;

    for (i = 0; i < keyLength; i++) {
        key += characters.substr(Math.floor((Math.random() * charactersLength) + 1), 1);
    }

    return key;
}


keyGen(12)
"QEt9mYBiTpYD"
于 2017-04-19T18:38:28.287 回答
2

当我看到这个问题时,我想到了何时必须生成 UUID。我不能相信代码,因为我确信我在 stackoverflow 上找到了它。如果您不想要字符串中的破折号,请取出破折号。这是功能:

function generateUUID() {
    var d = new Date().getTime();
    var uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g,function(c) {
        var r = (d + Math.random()*16)%16 | 0;
        d = Math.floor(d/16);
        return (c=='x' ? r : (r&0x7|0x8)).toString(16);
    });
    return uuid.toUpperCase();
}

小提琴:http: //jsfiddle.net/nlviands/fNPvf/11227/

于 2015-01-19T18:26:58.397 回答
2
var randomString = function(length) {
  var str = '';
  var chars ='0123456789ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz'.split(
      '');
  var charsLen = chars.length;
  if (!length) {
    length = ~~(Math.random() * charsLen);
  }
  for (var i = 0; i < length; i++) {
    str += chars[~~(Math.random() * charsLen)];
  }
  return str;
};
于 2016-05-19T08:36:14.347 回答
1

这个函数应该给出一个任意长度的随机字符串。

function randString(length) {
    var l = length > 25 ? 25 : length;
    var str = Math.random().toString(36).substr(2, l);
    if(str.length >= length){
        return str;
    }
    return str.concat(this.randString(length - str.length));
}

我已经通过以下成功的测试对其进行了测试。

function test(){
    for(var x = 0; x < 300000; x++){
        if(randString(x).length != x){
            throw new Error('invalid result for len ' + x);
        }
    }
}

我选择 25 的原因是因为实际上返回的字符串Math.random().toString(36).substr(2, 25)长度为 25。这个数字可以根据需要更改。

此函数是递归的,因此调用具有非常大值的函数可能会导致Maximum call stack size exceeded. 通过我的测试,我能够获得长度为 300,000 个字符的字符串。

通过将字符串作为第二个参数发送给函数,可以将此函数转换为尾递归。我不确定 JS 是否使用 Tail 调用优化

于 2016-08-10T13:03:14.777 回答
1

很好很简单,而且不限于一定数量的字符:

let len = 20, str = "";
while(str.length < len) str += Math.random().toString(36).substr(2);
str = str.substr(0, len);
于 2017-05-16T14:52:38.360 回答
0

这是一个生成随机字符串字母表的简单代码。看看这段代码是如何工作的。

go(lenthOfStringToPrint);- 使用此函数生成最终字符串。

var letters = {
  1: ["q","w","e","r","t","y","u","i","o","p","a","s","d","f","g","h","j","k","l","z","x","c","v","b","n","m"],
  2: ["Q","W","E","R","T","Y","U","I","O","P","A","S","D","F","G","H","J","K","L","Z","X","C","V","B","N","M"]
},i,letter,final="";
random = (max,min) => {
  return Math.floor(Math.random()*(max-min+1)+min);
}
function go(length) {
  final="",letter="";
  for (i=1; i<=length; i++){
    letter = letters[random(0,3)][random(0,25)];
    final+=letter;
  }
  return final;
}
于 2021-05-08T08:28:16.973 回答
0

可以只使用 lodash uniqueId

    _.uniqueId([prefix=''])

生成唯一 ID。如果给出了前缀,则将 ID 附加到它上面。

于 2021-09-25T13:54:13.280 回答
-2

使用 md5 库:https ://github.com/blueimp/JavaScript-MD5

最短的方法:

md5(Math.random())

如果要将大小限制为 5:

md5(Math.random()).substr(0, 5)

于 2016-06-16T22:01:39.617 回答