43

最近,我尝试在 Python 中存储和读取文件中的信息,遇到了一个小问题:我想从文本文件中读取类型信息。从字符串到 int 或到 float 的类型转换非常有效,但是从字符串到类型的类型转换似乎是另一个问题。自然,我尝试了这样的事情:

var_type = type('int')

但是,type不用作强制转换,而是用作查找变量类型的机制,实际上就是str在这里。

我找到了一种方法:

var_type = eval('int')

但我通常会尽量避免使用eval或尽可能避免使用函数/语句exec。所以我的问题如下:是否有另一种pythonic(更具体的)方法将字符串转换为类型?

4

4 回答 4

56

我喜欢使用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>
于 2015-04-23T18:29:29.457 回答
15

你对你想要做什么有点困惑。类型,也称为类,是对象,就像 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)
于 2012-08-02T16:19:03.743 回答
9

为什么不直接使用查找表?

known_types = {
    'int': int,
    'float': float,
    'str': str
    # etc
}

var_type = known_types['int']
于 2012-08-02T15:58:51.447 回答
5

也许这就是你想要的,它只查看内置类型:

def gettype(name):
    t = getattr(__builtins__, name)
    if isinstance(t, type):
        return t
    raise ValueError(name)
于 2012-08-02T11:04:40.317 回答