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

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

詳細はこちら
Python 3.x

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

Q&A

解決済

1回答

3791閲覧

pythonのsuper(class,self)を説明してほしい。

snowshink

総合スコア140

Python 3.x

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

0グッド

1クリップ

投稿2019/05/30 08:25

###super()の動き(仕様)を説明してほしいです。
###経緯
多重継承すると、継承順で一番先にメソッドを定義している
スーパークラスのメソッドしか呼び出されないので、
継承しているクラスすべてのオーバーライドしているメソッドを呼び出す方法を調べたら
super()を下記のように使えばいいということがわかりましたが、いまいちsuper()の仕様がわかりません。
##試してみたコード

python

1class greeting: 2 country=None 3 def morning(self):pass 4 5class jp_greeting(greeting): 6 def __init__(self): 7 self.country="japan" 8 def morning(self): 9 print("おはよう.") 10 super(jp_greeting,self).morning() 11 12class en_greeting(greeting): 13 def __init__(self): 14 self.country="america" 15 def morning(self): 16 print("good morning.") 17 super(en_greeting,self).morning() 18 19class de_greeting(greeting): 20 def __init__(self): 21 super(de_greeting,self).country="germany" 22 def morning(self): 23 print("guten tag.") 24 25class myGreeting(jp_greeting,en_greeting,de_greeting): 26 def __init__(self): 27 self.country="" 28 def morning(self): 29 super(myGreeting,self).morning() 30myGreeting().mornig()

##試したこと
0. 最後の呼び出しをsuper(myGreeting,myGreeting).mornig(myGreeting())にしても、得られる結果は同じでした。
0. en_greettingクラスのsuper(en_greeting,self).morning()super(jp_greeting,self).morning()にしたら永続ループしました。
0. 今度はsuper(de_greeting,self).morning()に変えたら、de_greetingのmorning()は呼び出されませんでした。

###予想
uper(superClass,class)はclassでsuperClassの次に継承されているクラスを指定する。
だから、superClassはメソッドを呼び出しているクラスにすると継承した順に呼び出される。

でも、これでは1.の動きを説明できません。

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

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

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

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

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

m.ts10806

2019/05/30 08:26

回答依頼いただいて申し訳ないのですがpythonは全く専門外なのでご期待にはそえません。
snowshink

2019/05/30 08:38

分かりました。すみません
guest

回答1

0

ベストアンサー

予想

super(superClass,class)はclassでsuperClassの次に継承されているクラスを指定する。

だから、superClassはメソッドを呼び出しているクラスにすると継承した順に呼び出される。

継承した順が「method resolution order(以下MROと略記)の順番」という意味と捉えるなら考え方自体は概ねあっているといえるでしょう。しかし

でも、これでは1.の動きを説明できません。

と言っておられることからsuperの引数についての正確な仕様を見落としておられるようです。正確な仕様は仕様書を見て把握しましょう。特にsuperのように若干トリッキーなものだと仕様書をみずに「例だけ見て正確な仕様を掴もうとする」のは難しいと思います。

(1) Python公式ドキュメント
Python 標準ライブラリ > 組み込み関数 > super([type[, object-or-type]])
https://docs.python.org/ja/3.7/library/functions.html#super
(2) オンラインヘルプ
help(super)およびIPython上でのsuper?

Pythonではクラスも型も同義ですが以下クラスと表記します。

myGreetingのMROは

myGreeting
jp_greeting
en_greeting
de_greeting
greeting
object

のようになります。これはmyGreeting.__mro__を見ると確認できます。これについては質問者さんも把握しておられると思います。

さて、質問者さんの予測に不足している点はsuperの第二引数はクラスではなくてもよく任意のオブジェクト(なんらかのクラスのインスタンス)が指定できることです。クラスでないものを指定した場合、そのオブジェクトのクラスがMROを決めます。

それゆえ

super(myGreeting, myGreeting).greeting(...)
と、(myGreetingのインスタンスである)selfを用いて
super(myGreeting, self).greeting(...)
とした場合のどちらでも同じMROとなるので.greeting(...)がjp_greeting.greetingになるのです。


ひょっとすると質問者さんの疑問は

super(myGreeting, myGreeting).greeting()
super(myGreeting, self).greeting(self)

のようにgreetingの引数にselfの指定有無がかわる点かも知れませんが、それはsuperの仕様というよりはPythonのメソッドの仕組みで用いられているbounded methodと同様の話と考えるとよいと思います。質問者さんの疑問がここにあるかどうかはっきりしないので、正確な動作を長々説明するのを止して大雑把な説明をしますと、

例1:

python

1class A: 2 def m(self): 3 pass 4 5a = A() 6a.m() # このように記述すると第一引数は指定しなくてもaになる 7A.m(a) # このように記述すると第一引数を明示しなくてはならなくなる

上でa.mの結果はbounded methodとなりメソッドのレシーバーがaの値であることをbounded methodオブジェクト自体が記憶しています。一方A.mと記述するとA.mの結果は単なる関数であり、mを定義したときの仮引数の通りに実引数を渡さないといけないことになります。

super(aClass, self).mとすると解決されたメソッドmがbounded methodとして返され
super(aClass, selfClass).mとすると解決されたメソッドは関数として返されます。要は例1と同じことと捉えることができるわけです。


追記: 前述のsuperの大雑把な説明はインスタンスメソッド内においてのものです。classmethodでもそうあってほしいとおりに動作しますが、staticmethodではsuperの引数を省略できずsuper(__class__, __class__)などと書かねばなりません。

投稿2019/05/30 19:25

編集2019/05/31 23:40
KSwordOfHaste

総合スコア18402

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問