__getitem__ 메서드 이해


140

나는 대부분의 문서를 읽었습니다. __getitem__파이썬 있는 봤지만 여전히 그 의미를 파악할 수 없습니다.

그래서 내가 이해할 수있는 것은 다음 __getitem__과 같은 호출을 구현하는 데 사용 된다는 것입니다.self[key] 입니다. 그러나 그것의 용도는 무엇입니까?

이런 식으로 정의 된 파이썬 클래스가 있다고 가정 해 봅시다.

class Person:
    def __init__(self,name,age):
        self.name = name
        self.age = age

    def __getitem__(self,key):
        print ("Inside `__getitem__` method!")
        return getattr(self,key)

p = Person("Subhayan",32)
print (p["age"])

예상대로 결과가 반환됩니다. 하지만 __getitem__애초에 왜 사용 합니까? 나는 또한 파이썬이__getitem__ 내부적으로 . 하지만 왜 그렇게합니까?

누군가 이것을 더 자세히 설명해 주시겠습니까?


이것은 한 가지 예제 사용에 관심이있을 수 있습니다. dict를 올바르게 하위 클래스 화하고 getitemsetitem을
roganjosh

4
__getitem__예제에서 의 사용은별로 의미가 없지만 .NET을 사용하는 기존 코드와 함께 작동해야하는 사용자 지정 목록 또는 사전과 같은 클래스를 작성해야한다고 상상해보십시오 []. 그것이 __getitem__유용한 상황 입니다.
피터 Witvoet

답변:


158

Cong Ma는 무엇을 __getitem__ 는 용도를 주지만 유용 할 수있는 예를 들어 보겠습니다. 건물을 모델링하는 클래스를 상상해보십시오. 건물의 데이터에는 각 층을 차지하는 회사에 대한 설명을 포함하여 여러 속성이 포함됩니다.

사용하지 않으면 __getitem__다음과 같은 클래스가 생깁니다.

class Building(object):
     def __init__(self, floors):
         self._floors = [None]*floors
     def occupy(self, floor_number, data):
          self._floors[floor_number] = data
     def get_floor_data(self, floor_number):
          return self._floors[floor_number]

building1 = Building(4) # Construct a building with 4 floors
building1.occupy(0, 'Reception')
building1.occupy(1, 'ABC Corp')
building1.occupy(2, 'DEF Inc')
print( building1.get_floor_data(2) )

그러나 우리는 __getitem__(그리고 그에 상응하는 __setitem__) Building 클래스를 'nicer'로 사용하도록 만들 수 있습니다.

class Building(object):
     def __init__(self, floors):
         self._floors = [None]*floors
     def __setitem__(self, floor_number, data):
          self._floors[floor_number] = data
     def __getitem__(self, floor_number):
          return self._floors[floor_number]

building1 = Building(4) # Construct a building with 4 floors
building1[0] = 'Reception'
building1[1] = 'ABC Corp'
building1[2] = 'DEF Inc'
print( building1[2] )

이렇게 사용하는지 여부는 __setitem__실제로 데이터를 추상화 할 계획에 따라 다릅니다.이 경우 건물을 바닥 컨테이너로 취급하기로 결정했습니다 (또한 건물에 대한 반복기를 구현할 수 있으며 슬라이스 기능까지도 구현할 수 있습니다. -즉, 한 번에 두 개 이상의 층 데이터를 가져옵니다. 필요한 항목에 따라 다릅니다.


15
답을 여러 번 읽은 후에 만 ​​배운 것을 공유하기 위해 getitem이 있으면 해당 함수를 명시 적으로 호출 할 필요가 없습니다. 그가 자신을 호출 할 때 building1[2]내부적으로 getitem을 호출합니다. 따라서 @ tony-suffolk-66이 만드는 요점은 단순히 objectname [variablename]을 호출하여 런타임 중에 클래스의 모든 속성 / 변수를 검색 할 수 있다는 것입니다. 처음에는 명확하지 않았기 때문에 이것을 명확히하고 누군가에게 도움이되기를 바라며 여기에 작성하십시오. 삭제하면 중복하십시오
mithunpaul

3
@mithunpaul 객체 [인덱스] 표기법은 클래스의 속성 / 변수 / 속성을 가져 오는 데 사용되지 않습니다. 컨테이너 객체에 대한 인덱싱입니다. 예를 들어 부모가 자식 목록을 유지하는 부모에서 자식 객체를 검색합니다. 내 예에서-Building 클래스는 컨테이너 (이 경우 Floor 이름)이지만 Floor 클래스의 컨테이너 클래스 일 수 있습니다.
Tony Suffolk 66

지원하지 않는 경우를 제외하고는 len()다음을 받게됩니다 TypeError.TypeError: object of type 'Building' has no len()
Ciasto piekarz

len (및 반복 등과 같은 다른 기능)을 지원하는 것은 내 예제의 목적이 아닙니다. dunder_len 메서드를 구현하는 것은 간단합니다.
Tony Suffolk 66

@ TonySuffolk66 : ____len____가 ____getitem____이 반복되는 예제에서 인덱스 (바닥)에 대한 반복 가능 여부를 결정하는 것이 맞습니까?
Alex

73

[]키 또는 인덱스 항목을 취득하기위한 구문은 구문 설탕입니다.

a[i]Python 호출 을 평가할 때 a.__getitem__(i)(또는 type(a).__getitem__(a, i), 그러나이 구분은 상속 모델에 관한 것이며 여기서는 중요하지 않습니다). 의 클래스 a가이 메서드를 명시 적으로 정의하지 않더라도 일반적으로 조상 클래스에서 상속됩니다.

모든 (Python 2.7) 특수 메서드 이름과 해당 의미는 https://docs.python.org/2.7/reference/datamodel.html#special-method-names에 나열되어 있습니다.


8

매직 메소드 __getitem__는 기본적으로 목록 항목, 사전 항목, 배열 요소 등에 액세스하는 데 사용됩니다. 인스턴스 속성의 빠른 조회에 매우 유용합니다.

여기에서는 'name', 'age'및 'dob'(생년월일)로 인스턴스화 할 수있는 예제 클래스 Person을 사용하여이를 보여줍니다. 이 __getitem__메소드는 이름 또는 성, 일, 월 또는 연도 등과 같은 인덱싱 된 인스턴스 속성에 액세스 할 수있는 방식으로 작성됩니다.

import copy

# Constants that can be used to index date of birth's Date-Month-Year
D = 0; M = 1; Y = -1

class Person(object):
    def __init__(self, name, age, dob):
        self.name = name
        self.age = age
        self.dob = dob

    def __getitem__(self, indx):
        print ("Calling __getitem__")
        p = copy.copy(self)

        p.name = p.name.split(" ")[indx]
        p.dob = p.dob[indx] # or, p.dob = p.dob.__getitem__(indx)
        return p

하나의 사용자 입력이 다음과 같다고 가정합니다.

p = Person(name = 'Jonab Gutu', age = 20, dob=(1, 1, 1999))

__getitem__메소드 의 도움으로 사용자는 색인화 된 속성에 액세스 할 수 있습니다. 예 :

print p[0].name # print first (or last) name
print p[Y].dob  # print (Date or Month or ) Year of the 'date of birth'

좋은 예! init에 여러 매개 변수가있을 때 getitem 을 구현하는 방법에 대해 모든 것을 검색하고 있었고 적절한 구현을 찾기 위해 고군분투하고 있었고 마침내 이것을 보았습니다! 찬성하고 감사합니다!
Rahul P
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.