인스턴스의 클래스 이름을 얻습니까?


1436

내가하고있는 함수가 인스턴스의 클래스가 파생 된 기본 클래스 인 경우 Python에서 객체의 인스턴스를 생성 한 클래스 이름을 어떻게 알 수 있습니까?

inspect 모듈 이 나를 도울 수 있다고 생각 했지만 여기서 원하는 것을 얻지 못하는 것 같습니다. 그리고 __class__회원 을 파싱 하지 않아서이 정보를 얻는 방법을 모르겠습니다.


클래스 변수 에서 정확히 무엇을 '파싱'하고 있습니까?
sykora

1
인스턴스가 속한 클래스의 최상위 이름 (모듈 이름 등이 없음)
Dan

나는 그 반대가 더 어려울 것이라고 생각한다 (메소드 / 함수가 정의 된 클래스를 얻는 것).
Alexey

답변:


1908

클래스 의 __name__속성 을 사용해 보셨습니까 ? 즉 type(x).__name__, 당신이 원하는 클래스 이름을 알려줄 것입니다.

>>> import itertools
>>> x = itertools.count(0)
>>> type(x).__name__
'count'

여전히 Python 2를 사용하는 경우 위의 방법은 새 스타일 클래스 에서만 작동 합니다 (Python 3 이상에서는 모든 클래스가 "새 스타일"클래스 임). 코드에서 일부 구식 클래스를 사용할 수 있습니다. 다음은 모두 작동합니다.

x.__class__.__name__

41
놀랍도록 간단합니다. 왜 dir(x.__class__)나열하지 않습니까?
cfi

43
사용하는 이유 __class__오버 type방법은? 그렇습니다 : type(x).__name__. 이중 밑줄 멤버를 직접 호출하지 않습니까? 그래도을 사용하는 방법을 알 수 없습니다 __name__.
jpmc26

25
__class__유형이 just이므로 이전 스타일 클래스와 호환 되려면 직접 사용해야 합니다 instance.
Quantum7

14
이것은 로깅, orm 및 프레임 워크 코드에서 종종 내장 된 typename (x)이 있어야하는데 충분히 사용됩니다. 사용자가 이름 을 얻기 위해 "guts"를 보도록 요구하는 것은 pythonic, IMO가 아닙니다.
Erik Aronesty

5
@ErikAronesty,def typename(x): return type(x).__name__
sleblanc 22시 07 분

390

수업 이름을 문자열로 원하십니까?

instance.__class__.__name__

6
또는 instance .__ class__ : 클래스 객체를 가져 오기 : D
Pencilcheck

8
이중 밑줄 속성을 사용하는 것이 안전합니까?
에두아르드 루카

5
@EduardLuca 왜 안전하지 않을까요? 내장 된 속성은 밑줄을 사용하므로 사용자가 작성한 코드와 충돌하지 않습니다.
JC Rocamonde

3
글쎄, 단일 밑줄은 메서드 / 속성이 개인이어야 함을 의미 / 제안한다는 것을 알고 있습니다 (Python AFAIK에서 개인 메서드를 실제로 구현할 수는 없지만).
에두아르 루카

28
@EduardLuca 시작 부분의 이중 밑줄은 시작 부분의 단일 밑줄과 비슷하지만 훨씬 더 "개인"( "python name mangling"을 찾습니다). 시작 과 끝의 이중 밑줄 은 다릅니다-파이썬 용으로 예약되어 있으며 개인용이 아닙니다 (예 : __init__ 및 __add__와 같은 마술 방법).
Leagsaidh Gordon

101

type ()?

>>> class A(object):
...    def whoami(self):
...       print type(self).__name__
...
>>>
>>> class B(A):
...    pass
...
>>>
>>>
>>> o = B()
>>> o.whoami()
'B'
>>>

이것은 클래스 멤버 와 동일 하지만,이 결과를 손으로 파싱해야합니다. 약간 성가신 일입니다.
Dan

8
난이게 좋아. 이런 식으로 기본 클래스에서 하위 클래스의 이름을 얻을 수 있습니다.
joctee 2016 년

6
또는 같은 행동을 취하는 self.__class__.__name__대신에 type(self).__name__. type()내가 알지 못하는 기능이 없다면 ?
andreb

2
당신이 사용하는 경우 type(item)목록 항목에 대한 결과가 될 것입니다 <type 'instance'>동안 item.__class__.__name__클래스 이름을 보유하고 있습니다.
Grochni

1
@Grochni가 언급 한 문제는 Python 2.x의 특정 클래스에만 관련이 있다고 생각합니다. 여기를 참조하십시오 : stackoverflow.com/questions/6666856/…
Nate CK

43
class A:
  pass

a = A()
str(a.__class__)

위의 샘플 코드 (대화식 인터프리터에서 입력시)는 속성이 호출 될 때 생성되는 '__main__.A'것과 반대로 'A'생성 __name__됩니다. 생성자 A.__class__에게 결과를 전달하면 str파싱이 처리됩니다. 그러나보다 명확한 것을 원하면 다음 코드를 사용할 수도 있습니다.

"{0}.{1}".format(a.__class__.__module__,a.__class__.__name__)

동일한 이름을 가진 클래스가 별도의 모듈에 정의 된 경우이 동작이 바람직 할 수 있습니다.

위에 제공된 샘플 코드는 Python 2.7.5에서 테스트되었습니다.


2.6.6에서 테스트되었습니다. 여전히 잘 작동합니다.
햄릿

29
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):
  ...

예:

>>> 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'
>>>

5
이것은 오래된 Python 2.x에만 해당됩니다. 3.x에서 bclass ()는 bclass (object)로 해석됩니다. 그럼에도 불구하고 새로운 클래스가 Python 2.2에 나타났습니다.
alcalde

16

좋은 질문.

다음은 누군가를 도울 수있는 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

13

또는 classmethod데코레이터를 사용할 수 있습니다 .

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

    def use_classname(self):
        return self.get_classname()

사용법 :

>>> A.get_classname()
'A'
>>> a = A()
>>> a.get_classname()
'A'
>>> a.use_classname()
'A'

10

특별한 __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'

내성 (Introspection)은 당신이 추구하는 것이기 때문에 항상 당신이 고려하고 싶을 수도 있습니다.


2
주목해야한다 __qualname__3.3 이상 파이썬입니다
FireAphis

10
그리고 나는 내 소프트웨어 "meth"의 어떤 것도 명명하지 않을 것입니다.
Bobort

__qualname__vs에 대한 관련 설명 __name__: stackoverflow.com/questions/58108488/what-is-qualname-in-python
ti7

0

인스턴스 클래스 이름을 가져 오려면

type(instance).__name__

또는

instance.__class__.__name__

둘 다 동일

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.