本文只討論在shell下生成隨機(jī)數(shù),不包括在C/C++等程序語言中的做法。
$RANDOM
RANDOM是bash的一個(gè)內(nèi)建函數(shù),會返回一個(gè)[0, 32676]內(nèi)的整數(shù)。
我們可以設(shè)置seed來得到相同的隨機(jī)數(shù),也可以為了盡量隨機(jī)而每次設(shè)置不一樣的seed。
$ RANDOM=10
$ echo $RANDOM
31855
$ RANDOM=$(date +%s)
$ echo $RANDOM
6676
生成[beg,end)范圍內(nèi)的隨機(jī)數(shù)可以這樣寫:
function random_range() {
local beg=$1
local end=$2
echo $((RANDOM % ($end - $beg) + $beg))
}
/dev/random & /dev/urandom
這兩個(gè)是 Linux上的字符設(shè)備,存儲著系統(tǒng)當(dāng)前運(yùn)行環(huán)境的實(shí)時(shí)數(shù)據(jù)。它可以看作是系統(tǒng)某個(gè)時(shí)刻的唯一值數(shù)據(jù),因此可以用作隨機(jī)數(shù)元數(shù)據(jù)。兩者的區(qū)別在于random是會阻塞的,而urandom不會。一般來說,用 od 命令即可:
$ od -An -N2 -i /dev/random
$ 26979
$ od -An -N2 -i /dev/urandom
$ 56345
也可以取其中幾行數(shù)據(jù)并計(jì)算其校驗(yàn)和:
$ head -10 /dev/urandom | cksum
134919525 2611
注意:不要使用cat,因?yàn)檩敵龈就2幌聛韣
seq + sort
sort 命令有一個(gè) -R 選項(xiàng),可以根據(jù)隨機(jī) hash 排序,那么我們就可以用 seq 命令先生成一個(gè)整數(shù)序列,然后用 sort 的 -R 選項(xiàng)處理取其中一行即可。例如:
seq $beg $end | sort -R | head -n1
值得注意的是,使用這種方法時(shí),要求的值域可以包含負(fù)數(shù)區(qū)域
shuf
shuf 和 'sort -R' 的作用類似,用來根據(jù)輸入生成隨機(jī)序列:
shuf -i $beg-$end -n1
uuid
可以直接訪問/proc/sys/kernel/random/uuid文件來獲取唯一的uuid,由于uuid比較長,如果想取中間的一段使用,比如最長的最后一段,可以用cut來獲?。?/p>
$ cat /proc/sys/kernel/random/uuid
4d8c0580-3286-437b-9ae2-707118506c00
$ cat /proc/sys/kernel/random/uuid | cut -f5 -d"-"
b8e195c7760e
最后
附上一個(gè)用前面uuid生成隨機(jī)數(shù)的方法來生成激活碼的shell腳本,COUNT和MIN_LEN變量控制總個(gè)數(shù)及單個(gè)激活碼的長度。
#!/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}