객체 지향 Python-고급 기능

여기에서는 Python이 제공하는 몇 가지 고급 기능을 살펴 보겠습니다.

클래스 디자인의 핵심 구문

여기서 우리는 파이썬이 어떻게 우리 클래스에서 연산자를 활용할 수 있는지 살펴볼 것입니다. 파이썬은 대체로 객체이고 메소드는 객체를 호출하며 이것은 편리한 구문으로 숨겨져 있어도 계속됩니다.

>>> var1 = 'Hello'
>>> var2 = ' World!'
>>> var1 + var2
'Hello World!'
>>>
>>> var1.__add__(var2)
'Hello World!'
>>> num1 = 45
>>> num2 = 60
>>> num1.__add__(num2)
105
>>> var3 = ['a', 'b']
>>> var4 = ['hello', ' John']
>>> var3.__add__(var4)
['a', 'b', 'hello', ' John']

그래서 우리 자신의 클래스에 매직 메소드 __add__를 추가해야한다면 우리도 그렇게 할 수 있습니다. 그렇게 해보자.

my_list라는 인수로 list를 취하는 생성자 __init__가있는 Sumlist라는 클래스가 있습니다.

class SumList(object):
   def __init__(self, my_list):
      self.mylist = my_list
   def __add__(self, other):
     new_list = [ x + y for x, y in zip(self.mylist, other.mylist)]

     return SumList(new_list)
   
   def __repr__(self):
      return str(self.mylist)

aa = SumList([3,6, 9, 12, 15])

bb = SumList([100, 200, 300, 400, 500])
cc = aa + bb # aa.__add__(bb)
print(cc) # should gives us a list ([103, 206, 309, 412, 515])

산출

[103, 206, 309, 412, 515]

그러나 다른 마술 방법에 의해 내부적으로 관리되는 많은 방법이 있습니다. 아래는 그들 중 일부입니다.

'abc' in var # var.__contains__('abc')
var == 'abc' # var.__eq__('abc')
var[1] # var.__getitem__(1)
var[1:3] # var.__getslice__(1, 3)
len(var) # var.__len__()
print(var) # var.__repr__()

기본 제공 유형에서 상속

클래스는 또한 내장 유형에서 상속 할 수 있습니다. 즉, 내장 유형에서 상속하고 거기에있는 모든 기능을 활용합니다.

아래 예제에서 우리는 사전에서 상속하고 있지만 __setitem__ 메소드 중 하나를 구현하고 있습니다. 이 (setitem)은 사전에 키와 값을 설정할 때 호출됩니다. 이것은 마법의 방법이므로 암시 적으로 호출됩니다.

class MyDict(dict):

   def __setitem__(self, key, val):
      print('setting a key and value!')
      dict.__setitem__(self, key, val)

dd = MyDict()
dd['a'] = 10
dd['b'] = 20

for key in dd.keys():
   print('{0} = {1}'.format(key, dd[key]))

산출

setting a key and value!
setting a key and value!
a = 10
b = 20

이전 예제를 확장 해 보겠습니다. 아래에서는 목록 인덱스를 다룰 때 더 잘 호출되는 __getitem__ 및 __setitem__이라는 두 가지 매직 메서드를 호출했습니다.

# Mylist inherits from 'list' object but indexes from 1 instead for 0!
class Mylist(list): # inherits from list
   def __getitem__(self, index):
      if index == 0:
         raise IndexError
      if index > 0:
         index = index - 1
         return list.__getitem__(self, index) # this method is called when

# we access a value with subscript like x[1]
   def __setitem__(self, index, value):
      if index == 0:
         raise IndexError
      if index > 0:
      index = index - 1
      list.__setitem__(self, index, value)

x = Mylist(['a', 'b', 'c']) # __init__() inherited from builtin list

print(x) # __repr__() inherited from builtin list

x.append('HELLO'); # append() inherited from builtin list

print(x[1]) # 'a' (Mylist.__getitem__ cutomizes list superclass
               # method. index is 1, but reflects 0!

print (x[4]) # 'HELLO' (index is 4 but reflects 3!

산출

['a', 'b', 'c']
a
HELLO

위의 예에서 Mylist에 세 개의 항목 목록을 설정하고 암시 적으로 __init__ 메서드가 호출되고 요소 x를 인쇄하면 세 개의 항목 목록 ([ 'a', 'b', 'c'])이 표시됩니다. 그런 다음이 목록에 다른 요소를 추가합니다. 나중에 인덱스 1과 인덱스 4를 요청합니다. 그러나 출력을 보면 우리가 요청한 (index-1)에서 요소를 얻습니다. 우리가 알고 있듯이 목록 인덱싱은 0에서 시작하지만 여기서 인덱싱은 1부터 시작합니다 (이것이 목록의 첫 번째 항목을 가져 오는 이유입니다).

명명 규칙

여기서 우리는 변수, 특히 전 세계 파이썬 프로그래머가 사용하는 개인 변수와 규칙에 사용할 이름을 살펴볼 것입니다. 변수는 개인용으로 지정되었지만 Python에는 개인 정보가 없으며 이것은 설계 상입니다. 다른 잘 문서화 된 언어와 마찬가지로 Python에는 강요하지는 않지만 홍보하는 이름 지정 및 스타일 규칙이 있습니다. “Guido van Rossum” the originator of Python, that describe the best practices and use of name and is called PEP8. Here is the link for this, https://www.python.org/dev/peps/pep-0008/

PEP는 Python 향상 제안을 나타내며 제안 된 변경 사항을 논의하기 위해 Python 커뮤니티에 배포 된 일련의 문서입니다. 예를 들어 모두 추천합니다.

  • 모듈 이름-all_lower_case
  • 클래스 이름 및 예외 이름-CamelCase
  • 전역 및 지역 이름-all_lower_case
  • 함수 및 메서드 이름-all_lower_case
  • 상수-ALL_UPPER_CASE

이것은 권장 사항 일 뿐이며 원하는 경우 다양 할 수 있습니다. 그러나 대부분의 개발자가 이러한 권장 사항을 따르므로 코드가 읽기 어렵습니다.

왜 컨벤션을 준수합니까?

우리가 얻을 수있는 PEP 권장 사항을 따를 수 있습니다.

  • 대다수의 개발자에게 더 친숙 함
  • 대부분의 코드 독자에게 더 명확합니다.
  • 동일한 코드 기반에서 작업하는 다른 기여자의 스타일과 일치합니다.
  • 전문 소프트웨어 개발자의 마크
  • 모두가 당신을 받아 들일 것입니다.

변수 이름 지정- '공개'및 '개인'

Python에서는 모듈과 클래스를 다룰 때 일부 변수 또는 속성을 전용으로 지정합니다. 파이썬에서는 객체 내부를 제외하고는 접근 할 수없는“Private”인스턴스 변수가 존재하지 않습니다. Private은 단순히 코드 사용자가 사용하기위한 것이 아니라 내부적으로 사용하기위한 것임을 의미합니다. 일반적으로 대부분의 Python 개발자는 규칙을 따르고 있습니다. 예를 들어 이름 앞에 밑줄이 붙습니다. _attrval (아래 예제)은 함수, 메서드 또는 데이터 멤버인지 여부에 관계없이 API 또는 Python 코드의 비공개 부분으로 처리되어야합니다. 다음은 우리가 따르는 명명 규칙입니다.

  • 공용 속성 또는 변수 (이 모듈의 임포터 또는이 클래스의 사용자가 사용하도록 의도 됨)-regular_lower_case

  • 개인 속성 또는 변수 (모듈 또는 클래스의 내부 사용)-_single_leading_underscore

  • 서브 클래 싱해서는 안되는 개인 속성-__double_leading_underscore

  • 마법 속성-__double_underscores__(사용하고 만들지 마십시오)

class GetSet(object):

   instance_count = 0 # public
   
   __mangled_name = 'no privacy!' # special variable

   def __init__(self, value):
      self._attrval = value # _attrval is for internal use only
      GetSet.instance_count += 1

   @property
   def var(self):
      print('Getting the "var" attribute')
      return self._attrval

   @var.setter
   def var(self, value):
      print('setting the "var" attribute')
      self._attrval = value

   @var.deleter
   def var(self):
      print('deleting the "var" attribute')
      self._attrval = None

cc = GetSet(5)
cc.var = 10 # public name
print(cc._attrval)
print(cc._GetSet__mangled_name)

산출

setting the "var" attribute
10
no privacy!