🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

関数

関数(ファンクション・メソッド・サブルーチンとも呼ばれる)は、はプログラムのコードの一部であり、ある特定のタスクを処理するように設計されたものです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

1回答

1360閲覧

pythonの**kwargsを適切に振り分けたい

slimemoss

総合スコア6

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

関数

関数(ファンクション・メソッド・サブルーチンとも呼ばれる)は、はプログラムのコードの一部であり、ある特定のタスクを処理するように設計されたものです。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2019/10/08 18:13

編集2019/10/08 18:17

目的

kwargsを複数の関数に対して、適切に振り分けたいです。

コードの例

mainfunc以外はパッケージ化されていて変更できないとします。

# 自由にいじれる関数 def mainfunc(**kwargs): func_a(**kwargs) func_b(**kwargs) # 以下、変更できない関数 def func_a(a1=0, **kwargs): func_a_inner(**kwargs) def func_a_inner(aa1=0): pass def func_b(b1=0, **kwargs): func_b_inner(**kwargs) def func_b_inner(bb1=0): pass

このようなコードに対して

mainfunc(a1=1, bb1=2)

のよう呼び出したいです。

そのままだと、以下のようなエラーが出ます。

mainfunc(a1=1)と呼び出すと、func_b_inner(a1=1)が呼ばれ、
**kwargsのない関数に、想定していないキーが来るため、エラーになります。

Traceback (most recent call last): File "a.py", line 22, in <module> mainfunc(a1=0) File "a.py", line 3, in mainfunc func_b(**kwargs) File "a.py", line 15, in func_b func_b_inner(**kwargs) TypeError: func_b_inner() got an unexpected keyword argument 'a1'

試したこと

  • inspect.signatureを用いた方法

関数の引数がどんなキーワードを待っているかを知ることができれば、うまく振り分けられるのですが、

param = inspect.signature(func_a).parameter print(param) >>> OrderedDict([('a1', <Parameter "a1=0">), ('xargs', <Parameter "**xargs">)])

上記の方法だと、aa1に関するkeywordのdictは得られません。
また、aa1を見つけるために、func_a_innerのsignatureを取っていくのはスマートではありません。
calleeを取得する方法は見つけることができませんでした。

  • jedi

emacsの補完では、calleeの引数ほdictまで出るので、そちらを参考にしようとしましたが、ちんぷんかんぷんでした。

  • ナイーブな方法

このような方法は取りたくありません

def mainfunc(func_a_kwargs, func_b_kwargs): func_a(**func_a_kwargs) func_b(**func_b_kwargs) main_func(func_a_kwargs={aa1:1}, func_b_kwargs={b1:2})
def mainfunc(**kwargs): func_a_key = {a1, aa1} func_b_key = {b1, bb1} ...

補足情報(FW/ツールのバージョンなど)

python 3.6
よろしくお願いいたします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

コードを実行するまでは内部でどの関数が呼ばれるかはわからないので、関数に kwargs を渡す段階でどのような引数を渡すかは決定できないですよね。

今回の例は、func_a 内では func_a_inner しか呼ばれていないですが、例えば、内部に条件分岐があるような場合なども考えられます。
どっちが呼ばれるかは func を呼び出す前の段階では判断できません。

python

1def func(a): 2 if a % 2 == 0: 3 even(b) 4 else: 5 odd(c)

変更できない関数の部分が外部のライブラリであるならば、API ドキュメントに受け取れる引数の記載があるので、複数の関数のパラメータを一緒に kwargs にわたすのではなく、関数ごとに明示的に指定する「ナイーブな方法」と書かれているものが一番自然な実装だと思います。

投稿2019/10/10 18:55

tiitoi

総合スコア21956

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

slimemoss

2019/10/11 17:07

ご意見ありがとうございます。 「API ドキュメントに受け取れる引数の記載があるので」という意見と、 振り分ける関数に同名のキーワードが使われている場合の対策のため、 ご回答の通り、関数ごとに明示的に指定する方法を取ることにしました。 誠にありがとうございました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問