本文只讨论在shell下生成随机数,不包括在C/C++等程序语言中的做法。
$RANDOM
RANDOM是bash的一个内建函数,会返回一个[0, 32676]内的整数。
我们可以设置seed来得到相同的随机数,也可以为了尽量随机而每次设置不一样的seed。
$ RANDOM=10
$ echo $RANDOM
31855
$ RANDOM=$(date +%s)
$ echo $RANDOM
6676
生成[beg,end)范围内的随机数可以这样写:
function random_range() {
local beg=$1
local end=$2
echo $((RANDOM % ($end - $beg) + $beg))
}
/dev/random & /dev/urandom
这两个是 Linux上的字符设备,存储着系统当前运行环境的实时数据。它可以看作是系统某个时刻的唯一值数据,因此可以用作随机数元数据。两者的区别在于random是会阻塞的,而urandom不会。一般来说,用 od 命令即可:
$ od -An -N2 -i /dev/random
$ 26979
$ od -An -N2 -i /dev/urandom
$ 56345
也可以取其中几行数据并计算其校验和:
$ head -10 /dev/urandom | cksum
134919525 2611
注意:不要使用cat,因为输出根本停不下来~
seq + sort
sort 命令有一个 -R 选项,可以根据随机 hash 排序,那么我们就可以用 seq 命令先生成一个整数序列,然后用 sort 的 -R 选项处理取其中一行即可。例如:
seq $beg $end | sort -R | head -n1
值得注意的是,使用这种方法时,要求的值域可以包含负数区域
shuf
shuf 和 'sort -R' 的作用类似,用来根据输入生成随机序列:
shuf -i $beg-$end -n1
uuid
可以直接访问/proc/sys/kernel/random/uuid文件来获取唯一的uuid,由于uuid比较长,如果想取中间的一段使用,比如最长的最后一段,可以用cut来获取:
$ cat /proc/sys/kernel/random/uuid
4d8c0580-3286-437b-9ae2-707118506c00
$ cat /proc/sys/kernel/random/uuid | cut -f5 -d"-"
b8e195c7760e
最后
附上一个用前面uuid生成随机数的方法来生成激活码的shell脚本,COUNT和MIN_LEN变量控制总个数及单个激活码的长度。
#!/bin/sh
FILE=code.txt
FILE_TMP=${FILE}.tmp
COUNT=100
MIN_LEN=10
GEN_COUNT=`expr ${COUNT} \* 2`
code_len=`cat /proc/sys/kernel/random/uuid | md5sum | cut -d' ' -f1 | wc -c`
if [ ${code_len} -gt ${MIN_LEN} ]; then
code_len=${MIN_LEN}
fi
for i in `seq ${GEN_COUNT}`
do
num=`cat /proc/sys/kernel/random/uuid | md5sum | cut -d' ' -f1`
echo ${num} | cut -b 1-${code_len} >> $FILE_TMP
done
sort $FILE_TMP | uniq | head -${COUNT} | shuf > ${FILE}
rm -f ${FILE_TMP}
网友评论