11

只是对 Python 中的全局值感到困惑,这里有两段代码

#gl.py

import cli

a = 1    
print "gl 1: %d %d" % (id(a), a)


def reset():
    global a
    a = 7
    print "reset 1: %d %d" % (id(a), a)


if __name__ == '__main__':
    cli.handler(reset)
    print "gl 2: %d %d" % (id(a), a)

cli代码

#cli.py

def handler(func):
    from gl import a
    print "cli 1: %d %d" % (id(a), a)
    func()
    print "cli 2: %d %d" % (id(a), a)

执行的结果是

$ python gl.py
gl 1: 150847672 1
gl 1: 150847672 1
cli 1: 150847672 1
reset 1: 150847600 7
cli 2: 150847672 1    #Why value doesn't change
gl 2: 150847600 7

这里我不明白“function reset()”执行后,全局值的结果在cli.py中没有改变(cli 2:150847672 1),但是回到gl.py,全局值确实改变了!!

4

3 回答 3

4

这里缺少两个概念

  • 全局变量是模块的全局变量,而不是跨模块的全局变量

参考: http: //legacy.python.org/doc/essays/ppt/hp-training/sld036.htm

参考:http ://docs.python.org/release/2.4/ref/global.html

  • 变量作为值导入,而不是通过引用导入

参考:https ://stackoverflow.com/a/3338357/977038

如果您需要跨模块共享全局变量,请参阅如何跨模块共享全局变量?

于 2013-03-06T06:32:38.890 回答
2

您在 cli 中导入的 gl 实际上是模块对象的副本。如果我们像这样更改您的代码:

#gl.py

import cli
import sys

a = 1    
print "gl 1: %d %d" % (id(a), a)
print "gl id on import: {0}".format(id(sys.modules[__name__]))


def reset():
    global a
    a = 7
    print "gl id in reset: {0}".format(id(sys.modules[__name__]))
    print "reset 1: %d %d" % (id(a), a)

def printa():
    print "gl: %d %d" % (id(a), a)

if __name__ == '__main__':
    cli.handler(reset)
    print "gl id in main: {0}".format(id(sys.modules[__name__]))
    print "gl 2: %d %d" % (id(a), a)

#cli.py

def handler(func):
    #from gl import a
    import gl
    print "gl id in cli: {0}".format(id(gl))
    print "cli 1: %d %d" % (id(gl.a), gl.a)
    func()
    print "cli 2: %d %d" % (id(gl.a), gl.a)
        gl.reset()
        print "cli 3: %d %d" % (id(gl.a), gl.a)

我们得到:

gl 1: 19056568 1
gl id on import: 140075849968728
gl 1: 19056568 1
gl id on import: 20004096
gl id in cli: 20004096
cli 1: 19056568 1
gl id in reset: 140075849968728
reset 1: 19056424 7
cli 2: 19056568 1
gl id in reset: 20004096
reset 1: 19056424 7
cli 3: 19056424 7
gl id in main: 140075849968728
gl 2: 19056424 7

所以当我们运行reset时,我们改变了引用

a -> 19056568

a -> 19056424

但只有一份 gl 副本。另一个(cli 中的那个)保留旧的参考。如果我们在 cli 中运行 gl.reset(),该副本上的引用会发生变化,我们会在 cli 中得到预期的变化。

于 2013-03-06T06:57:23.043 回答
2

您的gl模块被导入两次到两个不同的命名空间

尝试这个:

import sys
print sys.modules['__main__'].a
print sys.modules['gl'].a
于 2013-03-06T06:32:06.333 回答