web-dev-qa-db-ja.com

「mro()」は何をしますか?

_Django.utils.functional.py_で:

_for t in type(res).mro():  # <----- this
    if t in self.__dispatch:
        return self.__dispatch[t][funcname](res, *args, **kw)
_

mro()がわかりません。それは何をし、「mro」はどういう意味ですか?

114
zjm1126

従う...:

>>> class A(object): pass
... 
>>> A.__mro__
(<class '__main__.A'>, <type 'object'>)
>>> class B(A): pass
... 
>>> B.__mro__
(<class '__main__.B'>, <class '__main__.A'>, <type 'object'>)
>>> class C(A): pass
... 
>>> C.__mro__
(<class '__main__.C'>, <class '__main__.A'>, <type 'object'>)
>>> 

単一の継承がある限り、__mro__は、クラス、そのベース、そのベースのベースなどのobjectまでのタプルです(もちろん、新しいスタイルのクラスでのみ機能します) )。

さて、multiple inheritance ...:

>>> class D(B, C): pass
... 
>>> D.__mro__
(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <type 'object'>)

...また、__mro__では、クラスが重複せず、その先祖の後にクラスが来ないという保証があります。最初に同じレベルの多重継承で入るクラスを保存します(この例ではBとCのように)例)__mro__左から右にあります。

メソッドだけでなく、クラスのインスタンスで取得するすべての属性は、概念的に__mro__に沿って検索されるため、祖先の複数のクラスがその名前を定義している場合、これは属性の場所を示します-その名前を定義する__mro__の最初のクラス。

193
Alex Martelli

mro()はMethod Resolution Orderの略です。メソッドの検索順に、クラスの派生元の型のリストを返します。

mro()または__ mro __は、新しいスタイルクラスでのみ機能します。 python 3では、問題なく動作します。ただし、python 2では、これらのクラスはオブジェクトから継承する必要があります。

80
Ned Batchelder

これはおそらく解決の順序を示すでしょう。

class A(object):
    def dothis(self):
        print('I am from A class')

class B(A):
    pass

class C(object):
    def dothis(self):
        print('I am from C class')

class D(B, C):
    pass

d_instance= D()
d_instance.dothis()
print(D.mro())

そして応答は

I am from A class
[<class '__main__.D'>, <class '__main__.B'>, <class '__main__.A'>, <class '__main__.C'>, <class 'object'>]

ルールは深さ優先です。この場合、D、B、A、Cを意味します。

Pythonは通常、継承クラスを検索するときにdepth-first順序を使用しますが、2つのクラスが同じクラスから継承する場合、Pythonはmroからそのクラスの最初の言及を削除します。

9
Stryker

ダイヤモンドの継承では、解決の順序が異なります。

class A(object):
    def dothis(self):
        print('I am from A class')


class B1(A):
    def dothis(self):
        print('I am from B1 class')
    # pass


class B2(object):
    def dothis(self):
        print('I am from B2 class')
    # pass


class B3(A):
    def dothis(self):
        print('I am from B3 class')


# Diamond inheritance
class D1(B1, B3):
    pass


class D2(B1, B2):
    pass


d1_instance = D1()
d1_instance.dothis()
# I am from B1 class
print(D1.__mro__)
# (<class '__main__.D1'>, <class '__main__.B1'>, <class '__main__.B3'>, <class '__main__.A'>, <class 'object'>)


d2_instance = D2()
d2_instance.dothis()
# I am from B1 class
print(D2.__mro__)
# (<class '__main__.D2'>, <class '__main__.B1'>, <class '__main__.A'>, <class '__main__.B2'>, <class 'object'>)
1
Girish Gupta