学好eval() 可以让我们在使用python时候事半功倍,但是它也有一些弊端(在上篇《如何将字符串转换成字典》中讲到了)
例如:
1、字符串 lst_str="[1,2,3,4]"转换成列表就可以使用eval(lst_str),结果为列表类型[1,2,3,4]
2、同样字符串dict_str="{"name":"fandy","age":"18"}"使用eval(dict_str)转换成字典类型{"name":"fandy","age":"18"}
这里我们可以看下eval()的表达式:
eval
(expression[, globals[, locals]])
说明:
- expression:表达式,类型为字符串
- globals:全局变量,必须是字典
- locals:局部变量,可以是任何映射对象
- 表达式数据类型要一致,不然会报错
TypeError: can only concatenate str (not "int") to str
TypeError: unsupported operand type(s) for +: 'int' and 'str'
正确的写法:
a=1 g={"a":"aaa"} l={"b":"bbb"} res=eval("a+b+’hello'", g,l)# 结果为aaabbbhello
a=1 l={"a":50,"b":20} g={"a":10} res=eval("a+b+1", g,l)#结果为71
错误的写法:
TypeError: can only concatenate str (not "int") to str
a=1 g={"a":"aaa"} l={"b":"bbb"} res=eval("a+b+2", g,l)# 运行报错
TypeError: unsupported operand type(s) for +: 'int' and 'str'
a=1 l={"a":50,"b":20} g={"a":10} res=eval("a+b+’hello'", g,l)#运行报错
由此可见,eval()可以将其中的表达式去掉引号,转换成对应的类型
eval(“[1,2]”)---->列表[1,2]
eval(“{"a":1}”)---->字典{"a":1}
eval(“1+2”)---->int类型 的1+2
一、当只有expression表达式时
a=1 res=eval("a+3")#结果是4
由此可见,当只有表达式时,eval()直接执行里边的表达式,输出结果等价于print(1+3)
二、当同时存在表达式expression和全局变量globals时
a=1 g={"a":100} res=eval("a+3", g)#好结果为103
当同时存在表达式expression和全局变量globals时,取全局变量globals的数据,会屏蔽掉a=1,上面的例子中,a的数值取的是全局变量中的a=100,输出结果等价于print(100+3)
上面说了必须是字典,否则会报错:
TypeError: globals must be a real dict; try eval(expr, {}, mapping)
三、当同时存在表达式expression和全局变量globals、局部变量locals时
a=1 g={"a":100} l={"a":50,"b":20} res=eval("a+b+3", g,l)#结果为73
当同时存在表达式expression和全局变量globals、局部变量locals时,如果全局变量和局部变量有相同的key,则对应的key的value值取局部变量locals中的值;
上面的例子中,全局变量g和局部变量l中都有a,故表达式a+b+3中的a取局部变量l中的a=50,b只有局部变量有,故b=20,输出结果等价于print(50+20+3)