글
The Python 3.3.2 Tutorial (2)
Python Official Site의 Tutorial 에 대한 내용을 공부하면서 정리한 글.
기본적으로 이 글은 Tutorial의 각 챕터에 대한 간략한 설명과
노란색 박스 안에 있는 코드를 보는 것 만으로는 이해하기 어려운 부분이나,
매번 기억해 두어야 할 내용들 그리고 C/C++ 과는 좀 다른 부분
혹은 Python 2.x 버전과는 다른 부분 위주로 정리해 보려 함.
글의 순서는 원래 문서의 챕터에 따름.
이 글의 범위:
4. More Control Flow Tools
4. More Control Flow Tools
바로 이전 글에 보여준 while문 외에, 다른 control flow 문들에 대한 설명.
4.1. if Statements
기본적인 사용방법 예제. elif 부분이나 else 부분은 없어도 된다.
python에는 switch-case문은 없지만 if .. elif ... elif ... 문으로
switch-case문을 대체할 수 있다.
>>> x = int(input("Please enter an integer: "))
Please enter an integer: 42
>>> if x < 0:
... x = 0
... print('Negative changed to zero')
... elif x == 0:
... print('Zero')
... elif x == 1:
... print('Single')
... else:
... print('More')
...
More
4.2. for Statements
python의 for 문은 C 나 Pascal과는 조금 다르다.
항상 iterable 을 사용하여 반복을 한다.
기본적인 사용법은 아래와 같다.
>>> # Measure some strings:
... words = ['cat', 'window', 'defenestrate']
>>> for w in words:
... print(w, len(w))
...
cat 3
window 6
defenestrate 12
만약, for 루프 내에서 for 문의 iterating에 사용되는 데이터를 수정해야 한다면,
iteration에 사용되는 데이터는 아래와 같이 복사본을 사용해야 할 것이다.
>>> for w in words[:]: # Loop over a slice copy of the entire list.
... if len(w) > 6:
... words.insert(0, w)
...
>>> words
['defenestrate', 'cat', 'window', 'defenestrate']
4.3. The range() Function
일정한 순서의 숫자들을 반복(iterate)해야 한다면,
range()라는 내장(built-in) 함수를 사용하면 된다.
다음의 사용예를 보자.
>>> for i in range(5):
... print(i)
...
0
1
2
3
4
>>> a = ['Mary', 'had', 'a', 'little', 'lamb']
>>> for i in range(len(a)):
... print(i, a[i])
...
0 Mary
1 had
2 a
3 little
4 lamb
>>> print(range(10))
range(0, 10)
>>> list(range(5))
[0, 1, 2, 3, 4]
첫번째 예제를 보면 range(10)으로 리턴되는 object가 list처럼 사용되지만,
실제로는 그렇지 않다. 예제의 마지막 부분을 보면 단순히 range(10)을 한다고 하여
list가 리턴되지 않는다. 하지만 range(10)에 리턴되는 object는 iterable 하기 때문에,
for 문에서도 사용할 수 있었던 것이다.
그리고 list() 함수를 통해 list로 만들 수 있다.
4.4. break and continue Statements, and else Clauses on Loops
break 와 continue 문의 사용법은 C 와 동일하게 사용된다.
루프문도 else 절을 가질 수 있다. 이 else 절은 break 가 아닌
정상적인 경우로 for 나 while 루프가 종료된 경우,
else 절이 수행된다.
(for 문은 list 를 모두 소진하여 루프가 종료된 경우,
while 문은 조건이 false 가 되어 루프가 종료된 경우가 되겠다)
아래의 예제를 보자.
>>> for n in range(2, 10):
... for x in range(2, n):
... if n % x == 0:
... print(n, 'equals', x, '*', n//x)
... break
... else:
... # loop fell through without finding a factor
... print(n, 'is a prime number')
...
2 is a prime number
3 is a prime number
4 equals 2 * 2
5 is a prime number
6 equals 2 * 3
7 is a prime number
8 equals 2 * 4
9 equals 3 * 3
python에서는 들여쓰기(identation)가 매우 중요하다.
위의 예제의 경우도 else 의 위치가 바로 이전의 if 문 보다 왼쪽에 위치해 있기 때문에,
if 에 대한 else 가 아니라 for 루프에 대한 else 로 해석된다.
4.5. pass Statements
pass 문은 아무것도 하지 않는다.
문법적으로 statement가 필요하지만, 아무것도 하지 않으려 할 때 사용된다.
예를 들면 빈 클래스나 아무일도 하지 않는 함수를 만들 때 사용할 수 있겠다.
>>> class MyEmptyClass:
... pass
...
>>> def initlog(*args):
... pass # Remember to implement this!
...
4.6. Defining Functions
함수를 정의하는 방법에 대한 내용이다.
간단한 피보나치 수열을 구하는 함수예제를 살펴보자.
>>> def fib(n): # write Fibonacci series up to n
... """Print a Fibonacci series up to n."""
... a, b = 0, 1
... while a < n:
... print(a, end=' ')
... a, b = b, a+b
... print()
...
>>> # Now call the function we just defined:
... fib(2000)
0 1 1 2 3 5 8 13 21 34 55 89 144 233 377 610 987 1597
함수는 'def' 키워드로 시작하여 함수이름과 괄호로 묶인 파라메터 리스트가 나온다.
함수의 내용은 다음 줄 부터 시작되며 반드시 들여쓰기(ident)가 되어야 한다.
위 예제의 함수 본문의 첫 줄은 선택사항으로 함수에 대한 설명을 적는다.
이 문자열을 funtion's documentation string 혹은 docstring 이라 한다.
docstring을 잘 작성해 둔다면, API 문서 등을 생성하는 데 도움이 된다.
함수 내에서 사용되는 변수는 local symbol table에 저장되고,
함수를 벗어나게 되면 더 이상 사용할 수 없다.
global variable 을 사용하려 할 때는 C 처럼 바로 접근할 수 있는 것이 아니라,
'global' 키워드를 사용해야 한다.
>>> a=10
>>> def f():
a=20
>>> print (a)
10
>>> f()
>>> print (a)
10
>>> def f():
global a
a=20
>>> print (a)
10
>>> f()
>>> print (a)
20
함수의 파라메터는 호출되는 함수의 local symbol table에 저장되기 때문에,
arguments는 call by value를 사용하여 전달된다고 할 수 있겠다.
단, 이 value는 항상 object에 대한 reference이다.
C++ 로 치면 아래의 예제에서 함수 원형은 void f(&l); 정도 되겠다.
아래의 예제를 보면 list object의 복사본이 함수로 전달되는 것이 아니고,
실제 list object에 대한 reference가 넘어가기 때문에,
함수내에서 list의 내용을 변경하게 되면,
함수 호출 전/후로 list의 내용이 바뀌게 된다.
(각주: 그렇게 때문에 call by object reference 라고 하는게 더 정확한 설명이 될 것이다)
>>> a=[10,20,30]
>>> def f(l):
l[0]=50
>>> print(a)
[10, 20, 30]
>>> f(a)
>>> print(a)
[50, 20, 30]
함수도 다른 변수에 할당하여 다른 이름으로 사용할 수 있다.
마치 C 의 함수포인터 같은데... 아래 예제를 보자.
>>> fib
<function fib at 10042ed0>
>>> f = fib
>>> f(100)
0 1 1 2 3 5 8 13 21 34 55 89
함수는 return 문을 통해 값을 리턴할 수 있다.
return 문이 없는 함수의 리턴값은 None이다.(None은 built-in name이다.)
>>> def fib2(n): # return Fibonacci series up to n
... """Return a list containing the Fibonacci series up to n."""
... result = []
... a, b = 0, 1
... while a < n:
... result.append(a) # see below
... a, b = b, a+b
... return result
...
>>> f100 = fib2(100) # call it
>>> f100 # write the result
[0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89]
>>> def f():
pass
>>> print (f())
None
result.append(a) 는 result = result + [a] 와 동일하지만 더 효율적이다.
4.7. More on Defining Functions
함수의 arguments는 3가지 형태가 있으며, 섞어서 사용을 할 수가 있다.
4.7.1. Default Argument Values
arguments에 default value를 지정해 두어,
함수 호출 시 default value가 지정된 argument가 생략되면
default value를 사용하도록 한다. 다음의 예제를 보자.
def ask_ok(prompt, retries=4, complaint='Yes or no, please!'):
while True:
ok = input(prompt)
if ok in ('y', 'ye', 'yes'):
return True
if ok in ('n', 'no', 'nop', 'nope'):
return False
retries = retries - 1
if retries < 0:
raise IOError('refusenik user')
print(complaint)
위와 같이 정의된 함수라면 아래와 같이 호출 할 수 있다.
(위의 예제에서 if ~ in 의 사용법도 봐 두자.)
ask_ok('Do you really want to quit?')
ask_ok('OK to overwrite the file?', 2)
ask_ok('OK to overwrite the file?', 2, 'Come on, only yes or no!')
default value는 변수로 지정할 수도 있다.
함수가 정의되는 위치에서 evaluate되는 값을 사용한다.
예를 들어 아래 예제의 경우 '5'가 출력될 것이다.
i = 5
def f(arg=i):
print(arg)
i = 6
f()
경고: default value는 단 한번만 evaluate된다.
이것은 default value가 list나 dictionary 같은 mutable object인 경우,
다음과 같이 의도하지 않은 결과를 보여줄 수도 있다.
def f(a, L=[]):
L.append(a)
return L
print(f(1))
print(f(2))
print(f(3))
위의 예제는 아래와 같은 결과를 출력한다.
[1]
[1, 2]
[1, 2, 3]
만약 매번 f() 함수를 호출 할 때마다, L 이 초기화 되길 원한다면,
아래와 같이 사용해야 할 것이다.
def f(a, L=None):
if L is None:
L = []
L.append(a)
return L
4.7.2. Keyword Arguments
함수를 호출할 때, keyword arguments를 사용할 수 있다.
아래 예제를 보자.
def parrot(voltage, state='a stiff', action='voom', type='Norwegian Blue'):
print("-- This parrot wouldn't", action, end=' ')
print("if you put", voltage, "volts through it.")
print("-- Lovely plumage, the", type)
print("-- It's", state, "!")
parrot(1000) # 1 positional argument
parrot(voltage=1000) # 1 keyword argument
parrot(voltage=1000000, action='VOOOOOM') # 2 keyword arguments
parrot(action='VOOOOOM', voltage=1000000) # 2 keyword arguments
parrot('a million', 'bereft of life', 'jump') # 3 positional arguments
parrot('a thousand', state='pushing up the daisies') # 1 positional, 1 keyword
위와 같이 직접 키워드와 값을 지정하여 함수를 호출 할 수 있다.
또한, 임의의 갯수를 argument로 받거나 dictionary를 argument로 받을 수도 있다.
*arguments - arbitrary argument는 **keywords - dictionary argument 보다
위치가 앞에 와야한다. 아래 예제를 보면 어떻게 사용하는지 알 수 있다.
def cheeseshop(kind, *arguments, **keywords):
print("-- Do you have any", kind, "?")
print("-- I'm sorry, we're all out of", kind)
for arg in arguments:
print(arg)
print("-" * 40)
keys = sorted(keywords.keys())
for kw in keys:
print(kw, ":", keywords[kw])
cheeseshop("Limburger", "It's very runny, sir.",
"It's really very, VERY runny, sir.",
shopkeeper="Michael Palin",
client="John Cleese",
sketch="Cheese Shop Sketch")
위의 실행 결과는 아래와 같다.
-- Do you have any Limburger ?
-- I'm sorry, we're all out of Limburger
It's very runny, sir.
It's really very, VERY runny, sir.
----------------------------------------
client : John Cleese
shopkeeper : Michael Palin
sketch : Cheese Shop Sketch
4.7.3. Arbitrary Argument Lists
함수에서 임의의 갯수의 arguments를 받기위해 사용하는 방법인다.
이 arguments는 tuple 데이터 타입이 될 것이다.
일반적으로 이 arbitrary argument는 일반적인 argument의 가장 뒤에 위치하지만,
위에서 본 dictionary argument나 아래 예제에서 볼 keyword argument는
arbitrary argument 뒤에 위치할 수도 있다.
def write_multiple_items(file, separator, *args):
file.write(separator.join(args))
위는 일반적인 사용 법. 아래는 keyword argument와 함께 사용하는 예제.
>>> def concat(*args, sep="/"):
... return sep.join(args)
...
>>> concat("earth", "mars", "venus")
'earth/mars/venus'
>>> concat("earth", "mars", "venus", sep=".")
'earth.mars.venus'
4.7.4. Unpacking Argument Lists
list 나 tuple을 unpack 하여 함수인자로 넘길 수 있다.
같은 방식으로 dictionary로 unpack하여 인자로 넘길 수 있다.
아래 예제를 참고하자.
>>> list(range(3, 6)) # normal call with separate arguments
[3, 4, 5]
>>> args = [3, 6]
>>> list(range(*args)) # call with arguments unpacked from a list
[3, 4, 5]
>>> def parrot(voltage, state='a stiff', action='voom'):
... print("-- This parrot wouldn't", action, end=' ')
... print("if you put", voltage, "volts through it.", end=' ')
... print("E's", state, "!")
...
>>> d = {"voltage": "four million", "state": "bleedin' demised", "action": "VOOM"}
>>> parrot(**d)
-- This parrot wouldn't VOOM if you put four million volts through it. E's bleedin' demised !
4.7.5. Lambda Forms
많은 사람들의 요구에 의해 Lisp과 같은 functional programming language의 몇 가지 특징이
python에도 추가되었다. lambda 함수가 그 중 하나인데,
lambda 키워드로 작은 익명의 함수를 만들 수 있다.
lambda 형식은 function object가 필요한 어떤 곳이듯 사용될 수 있다.
그리고 문법적으로 항상 하나의 expression으로 제한된다.
아래의 예제를 참고하자.
>>> def make_incrementor(n):
... return lambda x: x + n
...
>>> f = make_incrementor(42)
>>> f(0)
42
>>> f(1)
43
>>> f = lambda a,b:a+b
>>> f(10,20)
30
4.7.6. Documentation Strings
docstring의 내용과 형식에 대한 몇 가지 규칙이 있다.
첫 번째 줄은 object의 목적에 대해 짧고 간결하게 적는다.
짧게 만들기 위해 object의 이름이나 타입을 적지 않는다.
(단, 함수의 이름이 함수의 동작을 설명하는 동사가 되는 경우는 제외한다.)
이 줄은 대문자로 시작하여 마침표로 끝난다.
만약 docstring이 여러 줄이라면, 함수의 요약이 나머지 설명과 시각적으로 분리되도록,
두 번째 줄은 빈 줄이어야 한다.
이 후 줄들은 object의 호출 규칙, side effect 등을 설명하는
하나 이상의 문단으로 이루어진다.
python 파서는 multi-line 문자열에 대해서는 들여쓰기(identation)를 잘라내지 않기 때문에,
필요하다면 들여쓰기를 없애는 작업이 필요할 것이다.
이러한 작업은 다음과 같은 규칙을 사용한다.
첫 번째 줄 이후 처음으로 나타나는 내용이 있는 줄(non-blank line)의 들여쓰기를
전체 docstring의 들여쓰기로 결정한다.
아래는 multi-line docstring 의 예제이다. 들여쓰기를 주의깊게 보라.
>>> def my_function():
... """Do nothing, but document it.
...
... No, really, it doesn't do anything.
... """
... pass
...
>>> print(my_function.__doc__)
Do nothing, but document it.
No, really, it doesn't do anything.
4.7.7. Function Annotations
이것은 완전한 선택사항이다.
python 이나 standard library 둘 다 이것을 사용하지 않는다.
자세한 것은 직접 tutorial을 참고 할 것.
4.8. Intermezzo: Coding Style
함수와는 관계 없으나 이제 더 길고 복잡한 python 코드를 작성하게 될텐데,
지금이 코딩 스타일에 대한 이야기를 하기에 좋은 타이밍이다.
PEP 8(Python Enhancement Proposals 8)는 대부분의 python projects가 준수하는
style guide를 나타낸다. 이것은 매우 가독성 있고 눈이 즐거운 코딩 스타일이다.
모든 python 개발자들은 반드시 읽어야 할 것이다.
여기는 가장 중요한 점만 뽑아놓았다.
- 4-space 들여쓰기를 사용하라. tab은 사용하지 말것.
4 spaces는 더 적은 들여쓰기와 더 큰 들여쓰기의 적절한 타협점이다.
tab은 혼란을 야기할 수 있으니 사용하지 않는 것이 좋다.
- 한 줄이 79 문자를 넘지 않도록 줄을 나누라.
이것은 작은 화면을 가진 유저를 위한 것이고,
더 큰 화면에서 여러 코드 파일을 동시에 볼 수 있게 해 준다.
- 함수와 클래스 그리고 함수 내부의 논리적 코드 블럭을 나누기 위해
빈 줄을 사용하라.
- 가능하다면, 주석은 해당 코드와 같은 라인에 두라.
- docstrings를 사용하라.
- 연산자 앞뒤와 콤마 뒤에는 공백을 두라. 하지만 괄호 바로 안쪽은 붙여쓰라.
a = f(1, 2) + g(3, 4)
- class 이름은 CamelCase 형태로, 함수는 lower_case_with_underscores 형태로 쓰라.
class methon 의 첫 번째 argument의 이름은 항상 self 를 사용하라.
- 소스 파일의 인코딩은 UTF-8을 사용하라. 여의치 않다면 ASCII 도 잘 동작할 것이다.
- 마찬가지로, 식별자로 non-ASCII 문자를 사용하지 마라.
'파이썬' 카테고리의 다른 글
The Python 3.3.2 Tutorial (1) (0) | 2013.08.07 |
---|---|
파이썬 GUI 툴킷 Tkinter 와 wxPython (0) | 2006.09.27 |
파이썬으로 파일 다운로드와 그 예외 처리 (0) | 2006.09.22 |
Python 공식 홈페이지의 Python Tutorial 중 기억해 둘 만한것... (0) | 2006.09.20 |
파이썬을 배워볼까나... (3) | 2006.09.20 |