从字符串定义变量

时间:2010-07-15 16:24:52

标签: python scope

我正在尝试在函数内部定义变量。 vars()显示变量已创建,但给出了NameError:exception。我做错了什么?

def a(str1):
    vars() [str1] =  1
    print vars()
    print b


a('b')

输出:

{'str1': 'b', 'b': 1}

异常:

NameError: global name 'b' is not defined

3 个答案:

答案 0 :(得分:4)

您正在调用未定义的行为。来自documentation of vars()

  

注意不应修改返回的字典:未定义对相应符号表的影响。

其他答案提供了可能的解决方案。

答案 1 :(得分:2)

您的代码适合我。也许你应该尝试另一种方法:

exec(str1 + '=1')

这将执行b=1

答案 2 :(得分:1)

如果您不理解为什么构造不起作用,那么下一个必须阅读您的代码的人也不会。如果你的意思是

b = 1   
你应该这么说。在这种情况下,vars()使您可以访问函数本地字典,因此您的代码等同于

def a():
   b = 1

其中ba的本地值,当它从a退出时超出范围时会消失。

过早优化是许多人试图猜测Python的根本

from itertools import izip
import timeit

import msw.wordlist

def zip_list(p):
    """construct a dictionary of length 100 from a list of strings"""
    return dict(zip(p[:100], p[100:]))

def izip_list(p):
    """as zip_list but avoids creating a new list to feed to dict()"""
    return dict(izip(p[:100], p[100:]))

def pass_list(p):
    """take 100 elements of a list and do nothing"""
    for i in p[:100]:
        pass

def exec_pass_list(p):
    """exec() a no-op 100 times"""
    for i in xrange(100):
        exec('pass')


# returns a list of 64'000 unique lowercase dictionary words for tests
words = msw.wordlist.Wordlist()
words.shuffle()
words = words[:200]
print 'words', words[:5], '...'

for func in ['zip_list', 'izip_list', 'pass_list', 'exec_pass_list']:
    t = timeit.Timer('%s(words)' % func,
            'from __main__ import words, %s' % func)
    print func, t.repeat(number=10**5)

产生:

words ['concatenated', 'predicament', 'shtick', 'imagine', 'stationing'] ...
zip_list [1.8603439331054688, 1.8597819805145264, 1.8571949005126953]
izip_list [1.5500969886779785, 1.5501470565795898, 1.5536530017852783]
pass_list [0.26778006553649902, 0.26837801933288574, 0.26767921447753906]
exec_pass_list [74.459679126739502, 75.221366882324219, 77.538936853408813]

无论OP试图做什么,我都没有费心去尝试工具,因为如果没有构建字典排序要快50倍,那么进一步测试就会变得愚蠢。