获取实例的类名?

Dan*_*Dan 1303 python introspection instanceof python-datamodel

如果我从中执行此操作的函数是实例的类派生的基类,如何找出在Python中创建对象实例的类的名称?

想到也许检查模块可能在这里帮助了我,但它似乎没有给我我想要的东西.如果没有解析__class__成员,我不知道如何获取这些信息.

syk*_*ora 1729

你有没有尝试过__name__这门课程的属性?ie type(x).__name__会给你这个班级的名字,我认为这就是你想要的.

>>> import itertools
>>> x = itertools.count(0)
>>> type(x).__name__
'count'
Run Code Online (Sandbox Code Playgroud)

此方法仅适用于新样式的类.您的代码可能会使用一些旧式类.以下适用于:

x.__class__.__name__
Run Code Online (Sandbox Code Playgroud)

  • 为什么在`type`方法上使用`__class__`?像这样:`type(x).__ name__`.是不是直接打电话给双下划线成员?但是,我无法看到使用`__name__`的方法. (41认同)
  • 非常简单.不知道为什么`dir(x .__ class __)`没有列出它? (37认同)
  • 你必须直接使用````__class__````来与旧式类兼容,因为它们的类型只是`````````. (25认同)
  • 这在日志记录,orm和框架代码中经常使用,它们确实应该是内置类型名称(x)...要求用户查看"guts"以获得__name__不是非常pythonic,IMO. (11认同)
  • @ErikAronesty,`def typename(x):return type(x).__ name__` (4认同)

tru*_*ppo 352

你想把类的名字作为字符串吗?

instance.__class__.__name__
Run Code Online (Sandbox Code Playgroud)

  • @EduardLuca一开始的双重下划线在开始时类似于单个下划线,但更加"私有"(查找"python name mangling").开头*和结尾*的双下划线是不同的 - 那些是为python保留的并且不是私有的(例如魔术方法,如\ _\_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ (24认同)
  • 使用双下划线属性是否安全? (7认同)
  • 或者实例.__ class__来获取类对象:D (5认同)
  • @EduardLuca为什么不安全?内置属性使用下划线,这样它们就不会与您编写的代码产生任何冲突 (5认同)
  • 我知道单个下划线意味着/建议方法/属性应该是私有的(尽管你不能在Python AFAIK中真正实现私有方法),我想知道是否也不是(某些)双下划线的情况. (2认同)
  • 或者,如果您从类内部执行此操作,则这将起作用 `self.__class__.__name__` (2认同)

GHZ*_*GHZ 90

type()?

>>> class A(object):
...    def whoami(self):
...       print type(self).__name__
...
>>>
>>> class B(A):
...    pass
...
>>>
>>>
>>> o = B()
>>> o.whoami()
'B'
>>>
Run Code Online (Sandbox Code Playgroud)

  • 我喜欢这一个.这样,在基类中可以获取子类的名称. (8认同)
  • 或者`self .__ class __.__ name__`而不是`type(self).__ name__`来获得相同的行为.除非有`type()`函数做了我不知道的事情? (6认同)
  • 如果你在列表项上使用`type(item)`,结果将是`<type'instal'>`而`item .__ class __.__ name__`包含类名. (2认同)

小智 37

class A:
  pass

a = A()
str(a.__class__)
Run Code Online (Sandbox Code Playgroud)

上面的示例代码(当在交互式解释器中输入时)将产生'__main__.A',而不是在调用属性时'A'产生的代码__name__.只需将结果传递A.__class__str构造函数,就可以为您处理解析.但是,如果您想要更明确的内容,也可以使用以下代码.

"{0}.{1}".format(a.__class__.__module__,a.__class__.__name__)
Run Code Online (Sandbox Code Playgroud)

如果您在单独的模块中定义了具有相同名称的类,则此行为更可取.

上面提供的示例代码在Python 2.7.5中进行了测试.


Pra*_*ath 26

type(instance).__name__ != instance.__class__.__name  #if class A is defined like
class A():
   ...

type(instance) == instance.__class__                  #if class A is defined like
class A(object):
  ...
Run Code Online (Sandbox Code Playgroud)

例:

>>> class aclass(object):
...   pass
...
>>> a = aclass()
>>> type(a)
<class '__main__.aclass'>
>>> a.__class__
<class '__main__.aclass'>
>>>
>>> type(a).__name__
'aclass'
>>>
>>> a.__class__.__name__
'aclass'
>>>


>>> class bclass():
...   pass
...
>>> b = bclass()
>>>
>>> type(b)
<type 'instance'>
>>> b.__class__
<class __main__.bclass at 0xb765047c>
>>> type(b).__name__
'instance'
>>>
>>> b.__class__.__name__
'bclass'
>>>
Run Code Online (Sandbox Code Playgroud)

  • 这仅适用于旧的Python 2.x. 在3.x中,bclass()将解析为bclass(object).即便如此,Python 2.2中出现了新的类. (4认同)

Rya*_*anN 14

好问题.

这是一个基于GHZ的简单示例,它可以帮助某些人:

>>> class person(object):
        def init(self,name):
            self.name=name
        def info(self)
            print "My name is {0}, I am a {1}".format(self.name,self.__class__.__name__)
>>> bob = person(name='Robert')
>>> bob.info()
My name is Robert, I am a person
Run Code Online (Sandbox Code Playgroud)


Lal*_*ara 11

您可以简单地使用__qualname__它代表函数或类的限定名称

例子:

>>> class C:
...     class D:
...         def meth(self):
...             pass
...
>>> C.__qualname__
'C'
>>> C.D.__qualname__
'C.D'
>>> C.D.meth.__qualname__
'C.D.meth'
Run Code Online (Sandbox Code Playgroud)

文档链接qualname

  • 这给出了类的名称而不是实例的类名 (2认同)

Yur*_*hko 8

或者你可以使用classmethod装饰器:

class A:
    @classmethod
    def get_classname(cls):
        return cls.__name__

    def use_classname(self):
        return self.get_classname()
Run Code Online (Sandbox Code Playgroud)

用法:

>>> A.get_classname()
'A'
>>> a = A()
>>> a.get_classname()
'A'
>>> a.use_classname()
'A'
Run Code Online (Sandbox Code Playgroud)


v.b*_*bak 7

获取实例类名:

type(instance).__name__
Run Code Online (Sandbox Code Playgroud)

或者

instance.__class__.__name__
Run Code Online (Sandbox Code Playgroud)

两者都是一样的


Jim*_*ard 6

除了获取特殊__name__属性外,您可能会发现自己需要给定类/函数的合格名称。这是通过获取类型来完成的__qualname__

在大多数情况下,它们将完全相同,但是在处理嵌套类/方法时,它们在输出中会有所不同。例如:

class Spam:
    def meth(self):
        pass
    class Bar:
        pass

>>> s = Spam()
>>> type(s).__name__ 
'Spam'
>>> type(s).__qualname__
'Spam'
>>> type(s).Bar.__name__       # type not needed here
'Bar'
>>> type(s).Bar.__qualname__   # type not needed here 
'Spam.Bar'
>>> type(s).meth.__name__
'meth'
>>> type(s).meth.__qualname__
'Spam.meth'
Run Code Online (Sandbox Code Playgroud)

由于自省是您所要追求的,因此始终可能需要考虑这一点。

  • 而且我会避免在软件“ meth”中命名任何东西。 (5认同)
  • 应该注意的是`__qualname__`适用于Python 3.3+ (2认同)