无法通过模块查看或修改 PYTHONHASHSEED 的值

Unable to see or modify value of PYTHONHASHSEED through a module

尝试使用以下方法查看 PYTHONHASHSEED 的值:

# (this is inside module_1.py)
os.environ['PYTHONHASHSEED'] 

引发 KeyError:

Traceback (most recent call last):
  File "/home/black/Dev/Projects/module_1.py", line 2, in <module>
    os.environ['PYTHONHASHSEED']
  File "/usr/lib/python3.4/os.py", line 631, in __getitem__
    raise KeyError(key) from None
KeyError: 'PYTHONHASHSEED'

打印 os.environ 的内容

for i in sorted(os.environ):
    print(i)

# prints:
# CLUTTER_IM_MODULE
# COMPIZ_BIN_PATH
# ...
# etc.

显示 PYTHONHASHSEED 不在里面,但我很确定它应该有一个值,因为 on Python 3.3 and greater, hash randomization is turned on by default.

问题:
为什么我不能以这种方式访问​​它的值,我如何通过 module_1.py?

访问它

可以 在 Python 脚本中设置 PYTHONHASHSEED,但它对 hash() 函数的行为没有影响 - 它需要设置在解释器启动前的解释器环境中。


如何使用纯Python

设置它的值

诀窍是将环境变量传递给子进程中的 Python 解释器。

import random
from subprocess import call

random.seed(37)
cmd = ['python', '-c', 'print(hash("abc"))']

for i in range(5):
    hashseed = bytes(random.randint(0, 4294967295))
    print('\nhashseed', hashseed)
    call(cmd, env={'PYTHONHASHSEED': hashseed})

输出

hashseed 2929187283
-972692480

hashseed 393430205
2066796829

hashseed 2653501013
1620854360

hashseed 3616018455
-599248233

hashseed 3584366196
-2103216293

您可以更改 cmd 列表,使其 运行 成为上面的 hashtest.py 脚本:

cmd = ['python', 'hashtest.py']

或者如果 hashtest.py 是可执行的,

cmd = './hashtest.py'

通过将 dict 作为 env 参数传递,我们替换了将传递给命令的默认环境。如果您需要访问那些其他环境变量,那么您应该在调用脚本中修改 os.environ,例如 os.environ['PYTHONHASHSEED'] = hashseed.

如何使用Bash

设置它的值

首先,我们有一个简短的 Bash 脚本 pyhashtest.bsh,它使用 RANDOM 环境变量作为 PYTHONHASHSEED 的种子。必须导出此变量,以便 Python 解释器可以看到它。然后我们 运行 我们的 Python 脚本 hashtest.py。我们循环执行此操作 5 次,因此我们可以看到使用不同的种子对哈希值有影响。

Python 脚本 hashtest.py 从环境中读取 PYTHONHASHSEED 并打印它以表明它具有我们期望的值。然后我们计算并打印一个短字符串的哈希值。

pyhashtest.bsh

#!/usr/bin/env bash

for((i=0; i<5; i++)); do
    n=$RANDOM
    echo "$i: Seed is $n"
    export PYTHONHASHSEED="$n"
    python hashtest.py
    echo
done

hashtest.py

#!/usr/bin/env python
import os

s = 'abc'
print('Hashseed is', os.environ['PYTHONHASHSEED'])
print('hash of s is', hash(s))

典型输出

0: Seed is 9352
Hashseed is 9352
hash of s is 401719638

1: Seed is 24945
Hashseed is 24945
hash of s is -1250185385

2: Seed is 17661
Hashseed is 17661
hash of s is -571990551

3: Seed is 24313
Hashseed is 24313
hash of s is 99658978

4: Seed is 21142
Hashseed is 21142
hash of s is -662114263

对于运行这些程序,将它们都保存到同一目录中,例如您运行 Python 脚本所在的常用目录。然后打开 Bash shell 并使用 cd 命令导航到该目录。

例如,如果您已将脚本保存到 /mnt/sda2/fred/python,那么您可以

cd /mnt/sda2/fred/python

接下来,使用此命令使 pyhashtest.bsh 可执行:

chmod a+x pyhashtest.bsh

然后运行它与

./pyhashtest.bsh

"hash randomization is turned on by default" 意味着(除其他外)即使没有 PYTHONHASHSEED 环境变量它也会打开 - 如果在环境中未指定哈希种子,则随机选择一个.此选择在 python 内部完成,os.environ 数组未更新。