330

我知道由于这样的混淆,我应该首先避免使用全局变量,但是如果我要使用它们,以下是使用它们的有效方法吗?(我试图调用在单独函数中创建的变量的全局副本。)

x = "somevalue"

def func_A ():
   global x
   # Do things to x
   return x

def func_B():
   x = func_A()
   # Do things
   return x

func_A()
func_B()

第二个函数使用的与使用和修改的全局副本的值是否x相同?在定义后调用函数时,顺序是否重要?xfunc_a

4

6 回答 6

489

如果您想简单地访问一个全局变量,您只需使用它的名称。但是,要更改其值,您需要使用global关键字。

例如

global someVar
someVar = 55

这会将全局变量的值更改为 55。否则它只会将 55 分配给局部变量。

函数定义列表的顺序无关紧要(假设它们不以某种方式相互引用),它们被调用的顺序很重要。

于 2012-05-14T17:38:03.323 回答
131

在 Python 范围内,对尚未在该范围内声明的变量的任何赋值都会创建一个新的局部变量,除非该变量在函数的前面声明为使用关键字 引用全局范围的变量global

让我们看一下伪代码的修改版本,看看会发生什么:

# Here, we're creating a variable 'x', in the __main__ scope.
x = 'None!'

def func_A():
  # The below declaration lets the function know that we
  #  mean the global 'x' when we refer to that variable, not
  #  any local one

  global x
  x = 'A'
  return x

def func_B():
  # Here, we are somewhat mislead.  We're actually involving two different
  #  variables named 'x'.  One is local to func_B, the other is global.

  # By calling func_A(), we do two things: we're reassigning the value
  #  of the GLOBAL x as part of func_A, and then taking that same value
  #  since it's returned by func_A, and assigning it to a LOCAL variable
  #  named 'x'.     
  x = func_A() # look at this as: x_local = func_A()

  # Here, we're assigning the value of 'B' to the LOCAL x.
  x = 'B' # look at this as: x_local = 'B'

  return x # look at this as: return x_local

实际上,您可以func_B使用命名的变量重写所有内容,x_local并且它的工作方式相同。

该顺序仅与您的函数执行更改全局 x 值的操作的顺序有关。因此,在我们的示例中,顺序无关紧要,因为func_B调用func_A. 在此示例中,顺序确实很重要:

def a():
  global foo
  foo = 'A'

def b():
  global foo
  foo = 'B'

b()
a()
print foo
# prints 'A' because a() was the last function to modify 'foo'.

请注意,global仅需要修改全局对象。您仍然可以从函数内访问它们而无需声明global. 因此,我们有:

x = 5

def access_only():
  return x
  # This returns whatever the global value of 'x' is

def modify():
  global x
  x = 'modified'
  return x
  # This function makes the global 'x' equal to 'modified', and then returns that value

def create_locally():
  x = 'local!'
  return x
  # This function creates a new local variable named 'x', and sets it as 'local',
  #  and returns that.  The global 'x' is untouched.

create_locally请注意和access_only--之间的区别,access_only尽管没有调用global,但它访问全局 x ,即使create_locally不使用global任何一个,它也会创建一个本地副本,因为它正在分配一个值。

这里的困惑是为什么你不应该使用全局变量。

于 2012-05-14T18:03:33.437 回答
23

正如其他人所指出的,global当您希望该函数能够修改全局变量时,您需要在函数中声明一个变量。如果您只想访问它,那么您不需要global.

更详细地说,“修改”的含义是:如果要重新绑定全局名称,使其指向不同的对象,则必须global在函数中声明该名称。

许多修改(变异)对象的操作不会重新绑定全局名称以指向不同的对象,因此它们都是有效的,无需在函数中声明名称global

d = {}
l = []
o = type("object", (object,), {})()

def valid():     # these are all valid without declaring any names global!
   d[0] = 1      # changes what's in d, but d still points to the same object
   d[0] += 1     # ditto
   d.clear()     # ditto! d is now empty but it`s still the same object!
   l.append(0)   # l is still the same list but has an additional member
   o.test = 1    # creating new attribute on o, but o is still the same object
于 2012-05-14T17:52:52.023 回答
20

您可以直接访问函数内部的全局变量。如果要更改该全局变量的值,请使用“global variable_name”。请参见以下示例:

var = 1
def global_var_change():
      global var
      var = "value changed"
global_var_change() #call the function for changes
print var

一般来说,这不是一个好的编程习惯。通过破坏命名空间逻辑,代码可能变得难以理解和调试。

于 2017-03-21T10:59:39.167 回答
9

这是一个让我印象深刻的案例,它使用全局作为参数的默认值。

globVar = None    # initialize value of global variable

def func(param = globVar):   # use globVar as default value for param
    print 'param =', param, 'globVar =', globVar  # display values

def test():
    global globVar
    globVar = 42  # change value of global
    func()

test()
=========
output: param = None, globVar = 42

我曾预计 param 的值为 42。惊喜。Python 2.7 在第一次解析函数 func 时评估了 globVar 的值。更改 globVar 的值不会影响分配给 param 的默认值。如下所示,延迟评估是我需要的。

def func(param = eval('globVar')):       # this seems to work
    print 'param =', param, 'globVar =', globVar  # display values

或者,如果你想安全,

def func(param = None)):
    if param == None:
        param = globVar
    print 'param =', param, 'globVar =', globVar  # display values
于 2016-06-01T15:16:35.087 回答
2

global当您希望更改分配给全局变量的值时,您必须使用该声明。

您不需要它从全局变量中读取。请注意,在对象上调用方法(即使它改变了该对象中的数据)不会改变持有该对象的变量的值(缺少反射魔法)。

于 2012-05-14T17:38:45.160 回答