我需要从一个shell脚本生成一个2000-65000之间的随机端口号。问题是$RANDOM是一个15位的数字,所以我被卡住了!

如果没有大小限制,PORT=$(($RANDOM%63000+2001))将很好地工作。

有没有人有一个例子,我可以这样做,也许从/dev/urandom提取一些东西,并得到它在一个范围内?


当前回答

如果你需要大于15位的范围,不要使用慢的不安全的过时的15位RANDOM,使用快速安全的32位SRANDOM。

SRANDOM自2021年bash 5.1推出以来可用。

Bash 5.1中值得注意的一个有趣的添加是新的SRANDOM变量。SRANDOM变量提供来自系统熵引擎的随机数据,不能被重新播种。特别是,SRANDOM变量提供了一个32位随机数,它依赖于getrandom/getentropy——如果必要的话,还可以回退到/dev/urandom或arc4random,甚至在此之后的另一个回退。”

来源:https://www.phoronix.com/news/gnu - bash - 5.1

看看bash中的RANDOM和SRANDOM有什么不同:

Bash中RANDOM和SRANDOM的区别

请随意改进这个答案。

其他回答

$RANDOM是0到32767之间的数字。您希望端口在2000到65000之间。这有63001个可能的端口。如果我们坚持在2000到33500之间使用$RANDOM + 2000的值,我们将覆盖31501个端口。如果我们抛硬币,然后有条件地在结果中加上31501,我们可以得到更多的端口,从33501到65001。然后,如果我们只删除65001,我们就可以获得所需的精确覆盖率,似乎所有端口的概率分布都是一致的。

random-port() {
    while [[ not != found ]]; do
        # 2000..33500
        port=$((RANDOM + 2000))
        while [[ $port -gt 33500 ]]; do
            port=$((RANDOM + 2000))
        done

        # 2000..65001
        [[ $((RANDOM % 2)) = 0 ]] && port=$((port + 31501)) 

        # 2000..65000
        [[ $port = 65001 ]] && continue
        echo $port
        break
    done
}

测试

i=0
while true; do
    i=$((i + 1))
    printf "\rIteration $i..."
    printf "%05d\n" $(random-port) >> ports.txt
done

# Then later we check the distribution
sort ports.txt | uniq -c | sort -r

PORT=$(($RANDOM%63000+2001))接近你想要的,我认为。

PORT=$(($RANDOM$RANDOM$RANDOM%63000+2001))绕过了困扰您的大小限制。由于bash不区分数字变量和字符串变量,因此这工作得非常好。“数字”$RANDOM可以像字符串一样连接,然后在计算中用作数字。神奇的!

在Bash中从10000000000 -999999999999范围内生成50个数字,并将它们保存到文件filename.csv中

shuf -i 100000000000-999999999999 -n 50 -o filename.csv

如果你需要大于15位的范围,不要使用慢的不安全的过时的15位RANDOM,使用快速安全的32位SRANDOM。

SRANDOM自2021年bash 5.1推出以来可用。

Bash 5.1中值得注意的一个有趣的添加是新的SRANDOM变量。SRANDOM变量提供来自系统熵引擎的随机数据,不能被重新播种。特别是,SRANDOM变量提供了一个32位随机数,它依赖于getrandom/getentropy——如果必要的话,还可以回退到/dev/urandom或arc4random,甚至在此之后的另一个回退。”

来源:https://www.phoronix.com/news/gnu - bash - 5.1

看看bash中的RANDOM和SRANDOM有什么不同:

Bash中RANDOM和SRANDOM的区别

请随意改进这个答案。

shuf -i 2000-65000 -n 1

享受吧!

编辑:范围是包括在内的。