switch (select) 構文のないPythonではその代わりに辞書を使うことがありますが,
その実行時間に関する注意です.
結論
- 関数自体を入れる:{ key: func }
- 短時間で済む
- 実行されるのはキーによって選択された関数のみ
- 関数の返り値を入れる:{ key: func() }
- 余計な時間がかかる
- 全ての関数が一旦実行され,その返り値が入る
関数の引数が異なる場合は,
- 予め関数の引数に
( *args, **kwargs )
を用意しておきその差を吸収する - 与える引数自体も辞書にしてしまう
- e.g.
dict_kwargs = { 1: { 'arg1': arg1 }, 2: { 'arg2': arg2 } }
- e.g.
という方法が考えられます.
例
以下の2つの関数を場合分けして実行する場合を考えます.
(実行時間が知りたいので遅くなるような書き方をしています.)
defmyfunc1():nx,ny=50,40arr=np.empty((nx,ny))foriyinrange(ny):forixinrange(nx):arr[ix,iy]=ix*ix+iy*iyreturnarrdefmyfunc2():nx,ny=50,40arr=np.empty((nx,ny))foriyinrange(ny):forixinrange(nx):arr[ix,iy]=1returnarr
その上で,以下の3通りの実行方法にかかる時間を計測しました(JupyterLabの%%timeit使用,下はmyfunc1のみ).
# 直接関数を実行
myfunc1()# { key: func } 型の辞書から実行
key=1dict_func={1:myfunc1,2:myfunc2}dict_func[key]()# { key: func() } 型の辞書から実行
dict_return={1:myfunc1(),2:myfunc2()}dict_return[key]
結果は以下の通りでした.{ key: func() } 型では,両方の関数が実行される時間に近い時間がかかっていることが分かります.
# 直接実行
1:488µs±26.7µsperloop(mean±std.dev.of7runs,1000loopseach)2:286µs±925nsperloop(mean±std.dev.of7runs,1000loopseach)# { key: func } 型
1:461µs±332nsperloop(mean±std.dev.of7runs,1000loopseach)2:298µs±22.1µsperloop(mean±std.dev.of7runs,1000loopseach)# { key: func() } 型
1:763µs±3.96µsperloop(mean±std.dev.of7runs,1000loopseach)2:771µs±31.6µsperloop(mean±std.dev.of7runs,1000loopseach)