最近,我试图在Python中存储和读取文件中的信息,并遇到了一个小问题:我想从文本文件中读取类型信息。从string到int或float的类型转换非常有效,但是从字符串到类型的类型转换似乎是另一个问题。当然,我尝试过这样的事情:
var_type = type('int')
但是,type
不是用作转换,而是用作查找变量类型的机制,实际上str
在这里。
我找到了一种方法:
var_type = eval('int')
但我通常尽量避免使用eval
或exec
这样的函数/语句。所以我的问题如下:是否有另一种pythonic(和更具体)的方法将字符串转换为类型?
答案 0 :(得分:28)
我喜欢使用locate
,它适用于内置类型:
>>> from pydoc import locate
>>> locate('int')
<type 'int'>
>>> t = locate('int')
>>> t('1')
1
......以及它在路径中可以找到的任何东西:
>>> locate('datetime.date')
<type 'datetime.date'>
>>> d = locate('datetime.date')
>>> d(2015, 4, 23)
datetime.date(2015, 4, 23)
...包括您的自定义类型:
>>> locate('mypackage.model.base.BaseModel')
<class 'mypackage.model.base.BaseModel'>
>>> m = locate('mypackage.model.base.BaseModel')
>>> m()
<mypackage.model.base.BaseModel object at 0x1099f6c10>
答案 1 :(得分:6)
你对你要做的事情感到有点困惑。类型,也称为类,是对象,就像python中的其他所有东西一样。当您在程序中编写int
时,您将引用一个名为int
的全局变量,该变量恰好是一个类。你要做的不是“将字符串转换为类型”,而是按名称访问内置变量。
一旦理解了这一点,就很容易看出解决方案:
def get_builtin(name):
return getattr(__builtins__, name)
如果确实想要将类型名称转换为类型对象,那么就是这样做的。我使用deque
进行广度优先的树遍历而不进行递归。
def gettype(name):
from collections import deque
# q is short for "queue", here
q = deque([object])
while q:
t = q.popleft()
if t.__name__ == name:
return t
else:
print 'not', t
try:
# Keep looking!
q.extend(t.__subclasses__())
except TypeError:
# type.__subclasses__ needs an argument, for whatever reason.
if t is type:
continue
else:
raise
else:
raise ValueError('No such type: %r' % name)
答案 2 :(得分:4)
为什么不使用查找表?
known_types = {
'int': int,
'float': float,
'str': str
# etc
}
var_type = known_types['int']
答案 3 :(得分:2)
也许这就是你想要的,它只关注内置类型:
def gettype(name):
t = getattr(__builtins__, name)
if isinstance(t, type):
return t
raise ValueError(name)