객체 지향 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!