少し時間が経ってしまいましたが前回からの続き。
最初はここから始まりました。
ここら辺から覚えることがちょっとしんどくなりそう。。。
制御フローとコード構造
関数定義
「def 関数名」で定義できる。
def say_something():
print('hi')
say_something()
hi
上から順番にスクリプトを読み込むので、関数定義される前に関数を呼び出そうとするとエラーになる。
say_something()
def say_something():
print('hi')
NameError: name 'say_something' is not defined
定義した関数はfunctionという型のオブジェクトになっていて、変数に入れて実行することもできる。
def say_something():
print('hi')
f = say_something
f()
hi
引数と返り値も指定できる。
def what_is_this(color):
if color == 'red':
return 'tomato'
if color == 'green':
return 'greeeeeeeeeeeeeeeeeen!!!!!!!!!'
else:
return '何かしらのあれ'
result = what_is_this('green')
print(result)
result = what_is_this('red')
print(result)
result = what_is_this('black')
print(result)
greeeeeeeeeeeeeeeeeen!!!!!!!!!
tomato
何かしらのあれ
関数の引数と返り値の宣言
関数の引数と返り値に型ヒントを指定することもできる。
あまり使われなていないらしい。
ただ、例えば引数、返り値ともにintを指定したとして、実際に関数を呼び出す際に引数に文字列を渡してもエラーとはならずに文字列で処理を実行してしまうので注意。ヒントでしかない。
def add_num(a: int, b: int) -> int:
return a + b
r = add_num(10, 20)
print(r)
# 文字列を渡しても実行できる
r = add_num('aと', 'bだよ')
print(r)
30
aとbだよ
位置引数とキーワード引数とデフォルト引数
位置引数とキーワード引数を合わせて使用できる。
どういう場面で組み合わせて使うかあまりイメージわかないなぁ…
def menu(entree, drink, dessert):
print(entree)
print(drink)
print(dessert)
menu('beef', dessert='ice', drink='beer')
beef
beer
ice
デフォルト引数も指定できる。
def menu(entree='tukemen', drink='tukejiru', dessert='gyoza'):
print(entree)
print(drink)
print(dessert)
menu()
menu(drink='beer')
tukemen
tukejiru
gyoza
tukemen
beer
gyoza
デフォルト引数で気を付けること
デフォルト引数にリストなどを指定するのはバグに繋がるのであまりよろしくないらしい。
# デフォルト引数でリストを指定した場合、スクリプトの関数が読み込まれた際に一回だけ生成される
# 新しいオブジェクトを生成するわけではないので、この関数のデフォルト引数を使用して複数回実行すると、同じリストに値が追加されていく。
# リストや辞書型はバグに繋がるのでデフォルト引数で与えるべきではないという暗黙の了解があるらしい
def test_func(x, l=[]):
l.append(x)
return l
y = [1, 2, 3]
r = test_func(100)
print(r)
print(id(r))
r = test_func(100)
print(r)
print(id(r))
y = test_func(100, y)
print(y)
print(id(y))
r = test_func(100)
print(r)
print(id(r))
[100]
30499336
[100, 100]
30499336
[1, 2, 3, 100]
30499400
[100, 100, 100]
30499336
それじゃデフォルト引数を使用するたびにリストを初期化して利用したい場合はどうするかというとこんな感じ。
def test_func(x, l=None):
if l is None:
# None(デフォルト引数)だったらリストを生成
l = []
l.append(x)
return l
y = [1, 2, 3]
r = test_func(100)
print(r)
print(id(r))
r = test_func(100)
print(r)
print(id(r))
y = test_func(100, y)
print(y)
print(id(y))
r = test_func(100)
print(r)
print(id(r))
[100]
36397640
[100]
44258824
[1, 2, 3, 100]
36397576
[100]
36397640
位置引数のタプル化
関数で引数の数が増えてくると書くのとかちょっと面倒。
そんなときは引数をタプルにできるんだよジョージィ。
# 「*変数名」で引数をまとめてくれる(タプルにしてくれる)
def say_something(*args):
# タプルになっている
print(args)
for arg in args:
print(arg)
say_something('Hi!', 'Mike', 'George')
('Hi!', 'Mike', 'George')
Hi!
Mike
George
通常の引数と組み合わせて使うこともできるみたい。
# wordは固定で使いたいけど、他は何個引数があるかわからないよジョージィな時に使えるらしい
def say_something(word, *args):
print('word =', word)
for arg in args:
print(arg)
say_something('Hi!', 'Mike', 'George', "It's Meeeeeeeeeeeeeeee!!!!!!!!!!")
# あまり使わないらしいけどこういった書き方もできる
t = ('Mike', 'Nancy')
say_something('Hey!', *t)
word = Hi!
Mike
George
It's Meeeeeeeeeeeeeeee!!!!!!!!!!
word = Hey!
Mike
Nancy
キーワード引数の辞書化
同じように辞書型にもできるらしい。
# 「**変数名」で辞書化できる
def menu(**kwargs):
print(kwargs)
for k, v in kwargs.items():
print(k, v)
menu(entree='beef', drink='coffee')
d = {
'entree': 'beef',
'drink': 'ice coffee',
'dessert': 'ice'
}
menu(**d)
{'entree': 'beef', 'drink': 'coffee'}
entree beef
drink coffee
{'entree': 'beef', 'drink': 'ice coffee', 'dessert': 'ice'}
entree beef
drink ice coffee
dessert ice
前項のタプルとかと一緒にも使える。
def menu(food, *args, **kwargs):
print(food)
print(args)
print(kwargs)
menu('banana', 'apple', 'orange', entree='beef', drink='coffee')
banana
('apple', 'orange')
{'entree': 'beef', 'drink': 'coffee'}
Docstrings
関数の説明記載できる。
関数内に「"""説明"""」と書いていく。
def example_func(param1, param2):
"""example_func document
Args:
param1 (int): The first parameter.
param2 (int): The second parameter.
Returns:
bool: The return value.
"""
print(param1)
print(param2)
return True
print(example_func.__doc__)
help(example_func)
example_func document
Args:
param1 (int): The first parameter.
param2 (int): The second parameter.
Returns:
bool: The return value.
Help on function example_func in module __main__:
example_func(param1, param2)
example_func document
Args:
param1 (int): The first parameter.
param2 (int): The second parameter.
Returns:
bool: The return value.
関数内関数
関数の中に関数を定義できる。
関数内のみで複数回使用するような場合に使用するらしい。
def outer(a, b):
def plus(c, d):
return c + d
r1 = plus(a, b)
r2 = plus(b, a)
print(r1 + r2)
outer(1, 2)
6
クロージャー
関数内の関数(inner)を返却値にすると
def outer(a, b):
def inner():
return a + b
return inner
# functionのオブジェクト情報が表示される
print(outer(1, 2))
<function outer.<locals>.inner at 0x0000000002A30D90>
このような感じでオブジェクト情報が表示される。
関数内の処理は実行されていない。
これを実行するにはこのようにする。
def outer(a, b):
def inner():
return a + b
return inner
# outer内のinnerは実行されていない
f = outer(1, 2)
# ここで初めて実行される
r = f()
# 実行された返り値が表示される
print(r)
3
具体的には↓みたいな使い方をするらしい。
# 面積求める関数
def circle_area_func(pi):
def circle_area(radius):
return pi * radius * radius
return circle_area
# πを2種類使いたい
ca1 = circle_area_func(3.14)
ca2 = circle_area_func(3.141592)
# それぞれのπで計算したい
print(ca1(10))
print(ca2(10))
314.0
314.1592
用途別で同じことをしたいときに使うのだろうか。
とりあえず何となくこういうものっていうことで覚えておいて次。
デコレーター
定義した関数の前後などに何かしら処理を付け加えたいとかっていうときに使えるらしい。
例えば計算する処理を実行する前後に開始と終了のメッセージを出すようにしたい。
# func実行前にstartとendを出力するデコレーター
def print_info(func):
def wrapper(*args, **kwargs):
print('start')
result = func(*args, **kwargs)
print('end')
return result
return wrapper
def add_num(a, b):
return a + b
# print_info内のwrapperが入って返ってくる
# wrapper内に記載されているfuncはadd_numが入っている
f = print_info(add_num)
# wrapperが実行され、add_numの結果が返ってくる
r = f(10, 20)
print(r)
start
end
30
これをこんな感じにも書ける。
def print_info(func):
def wrapper(*args, **kwargs):
print('start')
result = func(*args, **kwargs)
print('end')
return result
return wrapper
@print_info
def add_num(a, b):
return a + b
r = add_num(10, 20)
print(r)
デコレータを複数定義して使うこともできる。
記述順に上から実行されるので、順番は注意すること。
def print_more(func):
def wrapper(*args, **kwargs):
print('func:', func.__name__)
print('args:', args)
print('kwargs:', kwargs)
result = func(*args, **kwargs)
print('result:', result)
return result
return wrapper
def print_info(func):
def wrapper(*args, **kwargs):
print('start')
result = func(*args, **kwargs)
print('end')
return result
return wrapper
# デコレーター複数使うこともできる
# デコレーターの記述順に実行されていく
# print_info→print_more
@print_info
@print_more
def add_num1(a, b):
return a + b
# add_numと逆
# print_more→print_info
@print_more
@print_info
def add_num2(a, b):
return a + b
r1 = add_num1(10, 20)
print(r1)
print('##################')
r2 = add_num2(10, 20)
print(r2)
start
func: add_num1
args: (10, 20)
kwargs: {}
result: 30
end
30
##################
func: wrapper
args: (10, 20)
kwargs: {}
start
end
result: 30
30
正直よくわからないのでぐぐったら以下の記事がでてきた。
Pythonのデコレータについて
Pythonのデコレータを理解するための12Step
読んだけど理解しきれてないのでとりあえずこういうものっていうことで覚えておく。
ラムダ
頭文字が大文字だったり小文字だったりするリスト内の値を全て頭文字を大文字に直す処理があるとする。
# 頭が大文字だったり小文字だったりで揃っていない曜日のリスト
l = ['Mon', 'tue', 'Wed', 'Thu', 'fri', 'sat', 'Sun']
def change_words(words, func):
for word in words:
# 引数で渡されたfuncの返り値を出力する
print(func(word))
def sample_func(word):
return word.capitalize()
# sample_funcを渡して、リスト内の頭文字を大文字に変更して出力する
change_words(l, sample_func)
Mon
Tue
Wed
Thu
Fri
Sat
Sun
word.capitalize()のためだけにいちいち関数定義したくない。
別でword.lower()で全部小文字にしたい!とかもしたいけどそれだけのために関数定義したくない。
そんな時に
# 頭が大文字だったり小文字だったりで揃っていない曜日のリスト
l = ['Mon', 'tue', 'Wed', 'Thu', 'fri', 'sat', 'Sun']
def change_words(words, func):
for word in words:
# 引数で渡されたfuncの返り値を出力する
print(func(word))
# lambda~は先のsample_funcと同じ動きをする
change_words(l, lambda word: word.capitalize())
change_words(l, lambda word: word.lower())
Mon
Tue
Wed
Thu
Fri
Sat
Sun
mon
tue
wed
thu
fri
sat
sun
というように書ける。
関数の引数に関数を渡すときで、今回のように1行くらいの簡単な処理をする場合によく使われるらしい。
コード量も減るし見やすくなるしいいよね。
ジェネレーター
関数内に「yield 値」と書いたものをジェネレーターとして判断するらしい。
next(関数名)とするとyieldの値が返却される。
どこまで出力したか覚えてくれるようで、nextを繰り返し実行すると順番に処理される。
その次にyieldで定義したものがないのにnextで処理しようとするとこれ以上ないよとエラーになる。
def greeting():
yield 'Good morning'
yield 'Good afternoon'
yield 'Good night'
g = greeting()
# Good morningが出力される
print(next(g))
# Good afternoonが出力される
print(next(g))
# Good nightが出力される
print(next(g))
# ジェネレータで定義されているものがないのでエラーになる
print(next(g))
Good morning
Good afternoon
Good night
Traceback (most recent call last):
File "hoge/hoge.py", line 17, in <module>
print(next(g))
StopIteration
これもよくわからなかったのでぐぐったら記事出てきた。
Pythonのジェネレーターってなんのためにあるのかをなるべく分かりやすく説明しようと試みた
リスト内包表記
リストを生成する際に「[i for i in list]」というような感じでリストが作れる。
t = (1, 2, 3, 4, 5)
# このようにforで回してリストに追加して~とやるよりも
r = []
for i in t:
if i % 2 == 0:
r.append(i)
print(r)
# 上と同じ処理をこのように書ける
# こちらの方がメモリの消費が少なかったり処理早かったりするらしい
r = [i for i in t if i % 2 == 0]
print(r)
[2, 4]
[2, 4]
こんなのもリスト内包表記で書ける。
t = (1, 2, 3, 4, 5)
t2 = (5, 6, 7, 8, 9, 10)
r = []
for i in t:
for j in t2:
r.append(i * j)
print(r)
# 上と同じ処理をこのように書ける
r = [i * j for i in t for j in t2]
print(r)
[5, 6, 7, 8, 9, 10, 10, 12, 14, 16, 18, 20, 15, 18, 21, 24, 27, 30, 20, 24, 28, 32, 36, 40, 25, 30, 35, 40, 45, 50]
[5, 6, 7, 8, 9, 10, 10, 12, 14, 16, 18, 20, 15, 18, 21, 24, 27, 30, 20, 24, 28, 32, 36, 40, 25, 30, 35, 40, 45, 50]
これくらいなら問題ないけど、forを5個や6個とどんどん増やすとわけわからなくなるので、その場合は通常のfor文で処理を書くようにするといいそうな。
辞書包括表記
リスト内包表記と同じように辞書もできる。
w = ['mon', 'tue', 'wed']
f = ['coffee', 'milk', 'water']
d = {}
for x, y in zip(w, f):
d[x] = y
print(d)
# 上と同じ処理になる
d = {x: y for x, y in zip(w, f)}
print(d)
{'mon': 'coffee', 'tue': 'milk', 'wed': 'water'}
{'mon': 'coffee', 'tue': 'milk', 'wed': 'water'}
集合内包表記
こちらもリスト内包表記と同じようにできる。
s = set()
for i in range(10):
if i % 2 == 0:
s.add(i)
print(s)
s = {i for i in range(10) if i % 2 == 0}
print(s)
{0, 2, 4, 6, 8}
{0, 2, 4, 6, 8}
ジェネレーター内包表記
ジェネレーターにも内包表記がある。
def g():
for i in range(10):
yield i
g = g()
print(type(g))
print(next(g))
print(next(g))
print(next(g))
# 上と同じ処理になる
# ジェネレーターの内包表記は「()」になる。タプルにはならない。
g = (i for i in range(10))
print(type(g))
print(next(g))
print(next(g))
print(next(g))
# タプルの内包表記は「tuple()」にする。
t = tuple(i for i in range(10))
print(type(t))
print(t)
<class 'generator'>
0
1
2
<class 'generator'>
0
1
2
<class 'tuple'>
(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
名前空間とスコープ
ちょっとだけややこしい。
animal = 'cat'
def f():
# グローバル変数のanimalを出力
print(animal)
# ローカル変数のanimalを見ようとする
animal = 'dog'
print('after', animal)
f()
Traceback (most recent call last):
File "hoge/hoge.py", line 10, in <module>
f()
File "hoge./hoge.py", line 5, in f
print(animal)
UnboundLocalError: local variable 'animal' referenced before assignment
このようにエラーになる。
最初のprint行を削除すると、ローカル変数のanimalが定義されて、無事出力される。
それじゃグローバル変数の中身変えたいというときはどうするか。
「global 変数名」としてグローバル変数のanimalですよと宣言する。
animal = 'cat'
def f():
# グローバル変数のanimal
global animal
# グローバル変数のanimalにdogを入れる
animal = 'dog'
print('local', animal)
f()
print('global:', animal)
local dog
global: dog
ローカル変数を辞書で返してくれるlocals()、グローバル変数を辞書で返してくれるglobals()というものもある。
"""
Test Test #########################################
"""
animal = 'cat'
def f():
# ローカル変数のanimalにdogを入れる
animal = 'dog'
print('local:', locals())
f()
print('global:', globals())
local: {'animal': 'dog'}
global: {'__name__': '__main__', '__doc__': '\nTest Test #########################################\n', '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x0000000001D90EF0>, '__spec__': None, '__annotations__': {}, '__builtins__': <module 'builtins' (built-in)>, '__file__': 'C:/Users/xxxxxxxx/hogeProjects/hoge_programing/hoge.py', '__cached__': None, 'animal': 'cat', 'f': <function f at 0x0000000002F40D08>}
例外処理
「try 処理 except 例外時の処理」
exceptは例外毎に複数書ける。
Exceptionで例外出たら全部これで受け取るというような形はあまりよろしくない。
l = [1, 2, 3]
i = 5
try:
l[i]
except IndexError as ex:
print("Don't worry: {}".format(ex))
except NameError as ex:
print(ex)
print('last')
Don't worry: list index out of range
last
finallyもある。
l = [1, 2, 3]
i = 5
try:
l[i]
except IndexError as ex:
print("Don't worry: {}".format(ex))
except NameError as ex:
print(ex)
finally:
print('clean up')
whileと同じようにelseもある。
elseは例外が発生せずにtry文を抜けた場合にのみ実行される。
例外が発生した場合は実行されない。
l = [1, 2, 3]
try:
l[0]
except IndexError as ex:
print("Don't worry: {}".format(ex))
except NameError as ex:
print(ex)
else:
# 例外発生していないので実行される
print('done')
finally:
print('clean up1')
print('##################')
del l
try:
l[0]
except IndexError as ex:
print("Don't worry: {}".format(ex))
except NameError as ex:
print(ex)
else:
# 例外発生したので実行されない
print('done')
finally:
print('clean up2')
done
clean up1
##################
name 'l' is not defined
clean up2
独自例外の作成
「raise 例外名(メッセージ)」で例外発生させることができる。
# クラス名(継承元クラス)で継承できる
class UppercaseError(Exception):
pass
def check():
words = ['APPLE', 'orange', 'banana']
for word in words:
if word.isupper():
raise UppercaseError(word)
check()
Traceback (most recent call last):
File "hoge/hoge.py", line 12, in <module>
check()
File "hoge/hoge.py", line 10, in check
raise UppercaseError(word)
__main__.UppercaseError: APPLE
もちろんtryで独自例外を使える。
# クラス名(継承元クラス)で継承できる
class UppercaseError(Exception):
pass
def check():
words = ['APPLE', 'orange', 'banana']
for word in words:
if word.isupper():
raise UppercaseError(word)
try:
check()
except UppercaseError as ex:
print('This is my fault . Go next')
This is my fault . Go next
途中でペニーワイズが出てきたけどきっと眠かったか疲れてた。
続きはまた後日