导航:首页 > 编程语言 > pythonbaseclass

pythonbaseclass

发布时间:2025-04-06 07:24:50

㈠ 如何在python中使用static,class,abstract方法

方法在Python中是如何工作的

方法就是一个函数,它作为一个类属性而存在,你可以用如下方式来声明、访问一个函数:

Python

>>> class Pizza(object):
... def __init__(self, size):
... self.size = size
... def get_size(self):
... return self.size
...
>>> Pizza.get_size
<unbound method Pizza.get_size>

Python在告诉你,属性_get_size是类Pizza的一个未绑定方法。这是什么意思呢?很快我们就会知道答案:

Python

>>> Pizza.get_size()
Traceback (most recent call last):
File "<stdin>", line 1, in <mole>
TypeError: unbound method get_size() must be called with Pizza instance as first argument (got nothing instead)

我们不能这么调用,因为它还没有绑定到Pizza类的任何实例上,它需要一个实例作为第一个参数传递进去(Python2必须是该类的实例,Python3中可以是任何东西),尝试一下:

Python

>>> Pizza.get_size(Pizza(42))
42

>>> Pizza.get_size(Pizza(42))
42

太棒了,现在用一个实例作为它的的第一个参数来调用,整个世界都清静了,如果我说这种调用方式还不是最方便的,你也会这么认为的;没错,现在每次调用这个方法的时候我们都不得不引用这个类,如果不知道哪个类是我们的对象,长期看来这种方式是行不通的。

那么Python为我们做了什么呢,它绑定了所有来自类_Pizza的方法以及该类的任何一个实例的方法。也就意味着现在属性get_size是Pizza的一个实例对象的绑定方法,这个方法的第一个参数就是该实例本身。

Python

>>> Pizza(42).get_size
<bound method Pizza.get_size of <__main__.Pizza object at 0x7f3138827910>>
>>> Pizza(42).get_size()
42

和我们预期的一样,现在不再需要提供任何参数给_get_size,因为它已经是绑定的,它的self参数会自动地设置给Pizza实例,下面代码是最好的证明:

Python

>>> m = Pizza(42).get_size
>>> m()
42

更有甚者,你都没必要使用持有Pizza对象的引用了,因为该方法已经绑定到了这个对象,所以这个方法对它自己来说是已经足够了。

也许,如果你想知道这个绑定的方法是绑定在哪个对象上,下面这种手段就能得知:

Python

>>> m = Pizza(42).get_size
>>> m.__self__
<__main__.Pizza object at 0x7f3138827910>
>>> # You could guess, look at this:
...
>>> m == m.__self__.get_size
True

显然,该对象仍然有一个引用存在,只要你愿意你还是可以把它找回来。

在Python3中,依附在类上的函数不再当作是未绑定的方法,而是把它当作一个简单地函数,如果有必要它会绑定到一个对象身上去,原则依然和Python2保持一致,但是模块更简洁:

Python

>>> class Pizza(object):
... def __init__(self, size):
... self.size = size
... def get_size(self):
... return self.size
...
>>> Pizza.get_size
<function Pizza.get_size at 0x7f307f984dd0>

静态方法

静态方法是一类特殊的方法,有时你可能需要写一个属于这个类的方法,但是这些代码完全不会使用到实例对象本身,例如:

Python

class Pizza(object):
@staticmethod
def mix_ingredients(x, y):
return x + y

def cook(self):
return self.mix_ingredients(self.cheese, self.vegetables)

这个例子中,如果把_mix_ingredients作为非静态方法同样可以运行,但是它要提供self参数,而这个参数在方法中根本不会被使用到。这里的@staticmethod装饰器可以给我们带来一些好处:

Python不再需要为Pizza对象实例初始化一个绑定方法,绑定方法同样是对象,但是创建他们需要成本,而静态方法就可以避免这些。

Python

>>> Pizza().cook is Pizza().cook
False
>>> Pizza().mix_ingredients is Pizza.mix_ingredients
True
>>> Pizza().mix_ingredients is Pizza().mix_ingredients
True

可读性更好的代码,看到@staticmethod我们就知道这个方法并不需要依赖对象本身的状态。
可以在子类中被覆盖,如果是把mix_ingredients作为模块的顶层函数,那么继承自Pizza的子类就没法改变pizza的mix_ingredients了如果不覆盖cook的话。

类方法

话虽如此,什么是类方法呢?类方法不是绑定到对象上,而是绑定在类上的方法。

Python

>>> class Pizza(object):
... radius = 42
... @classmethod
... def get_radius(cls):
... return cls.radius
...
>>>
>>> Pizza.get_radius
<bound method type.get_radius of <class '__main__.Pizza'>>
>>> Pizza().get_radius
<bound method type.get_radius of <class '__main__.Pizza'>>
>>> Pizza.get_radius is Pizza().get_radius
True
>>> Pizza.get_radius()
42

无论你用哪种方式访问这个方法,它总是绑定到了这个类身上,它的第一个参数是这个类本身(记住:类也是对象)。

什么时候使用这种方法呢?类方法通常在以下两种场景是非常有用的:

工厂方法:它用于创建类的实例,例如一些预处理。如果使用@staticmethod代替,那我们不得不硬编码Pizza类名在函数中,这使得任何继承Pizza的类都不能使用我们这个工厂方法给它自己用。

Python

class Pizza(object):
def __init__(self, ingredients):
self.ingredients = ingredients

@classmethod
def from_fridge(cls, fridge):
return cls(fridge.get_cheese() + fridge.get_vegetables())

调用静态类:如果你把一个静态方法拆分成多个静态方法,除非你使用类方法,否则你还是得硬编码类名。使用这种方式声明方法,Pizza类名明永远都不会在被直接引用,继承和方法覆盖都可以完美的工作。

Python

class Pizza(object):
def __init__(self, radius, height):
self.radius = radius
self.height = height

@staticmethod
def compute_area(radius):
return math.pi * (radius ** 2)

@classmethod
def compute_volume(cls, height, radius):
return height * cls.compute_area(radius)

def get_volume(self):
return self.compute_volume(self.height, self.radius)

抽象方法

抽象方法是定义在基类中的一种方法,它没有提供任何实现,类似于Java中接口(Interface)里面的方法。

在Python中实现抽象方法最简单地方式是:

Python

class Pizza(object):
def get_radius(self):
raise NotImplementedError

任何继承自_Pizza的类必须覆盖实现方法get_radius,否则会抛出异常。

这种抽象方法的实现有它的弊端,如果你写一个类继承Pizza,但是忘记实现get_radius,异常只有在你真正使用的时候才会抛出来。

Python

>>> Pizza()
<__main__.Pizza object at 0x7fb747353d90>
>>> Pizza().get_radius()
Traceback (most recent call last):
File "<stdin>", line 1, in <mole>
File "<stdin>", line 3, in get_radius
NotImplementedError

还有一种方式可以让错误更早的触发,使用Python提供的abc模块,对象被初始化之后就可以抛出异常:

Python

import abc

class BasePizza(object):
__metaclass__ = abc.ABCMeta

@abc.abstractmethod
def get_radius(self):
"""Method that should do something."""

使用abc后,当你尝试初始化BasePizza或者任何子类的时候立马就会得到一个TypeError,而无需等到真正调用get_radius的时候才发现异常。

Python

>>> BasePizza()
Traceback (most recent call last):
File "<stdin>", line 1, in <mole>
TypeError: Can't instantiate abstract class BasePizza with abstract methods get_radius

混合静态方法、类方法、抽象方法

当你开始构建类和继承结构时,混合使用这些装饰器的时候到了,所以这里列出了一些技巧。

记住,声明一个抽象的方法,不会固定方法的原型,这就意味着虽然你必须实现它,但是我可以用任何参数列表来实现:

Python

import abc

class BasePizza(object):
__metaclass__ = abc.ABCMeta

@abc.abstractmethod
def get_ingredients(self):
"""Returns the ingredient list."""

class Calzone(BasePizza):
def get_ingredients(self, with_egg=False):
egg = Egg() if with_egg else None
return self.ingredients + egg

这样是允许的,因为Calzone满足BasePizza对象所定义的接口需求。同样我们也可以用一个类方法或静态方法来实现:

Python

import abc

class BasePizza(object):
__metaclass__ = abc.ABCMeta

@abc.abstractmethod
def get_ingredients(self):
"""Returns the ingredient list."""

class DietPizza(BasePizza):
@staticmethod
def get_ingredients():
return None

这同样是正确的,因为它遵循抽象类BasePizza设定的契约。事实上get_ingredients方法并不需要知道返回结果是什么,结果是实现细节,不是契约条件。

因此,你不能强制抽象方法的实现是一个常规方法、或者是类方法还是静态方法,也没什么可争论的。从Python3开始(在Python2中不能如你期待的运行,见issue5867),在abstractmethod方法上面使用@staticmethod和@classmethod装饰器成为可能。

Python

import abc

class BasePizza(object):
__metaclass__ = abc.ABCMeta

ingredient = ['cheese']

@classmethod
@abc.abstractmethod
def get_ingredients(cls):
"""Returns the ingredient list."""
return cls.ingredients

别误会了,如果你认为它会强制子类作为一个类方法来实现get_ingredients那你就错了,它仅仅表示你实现的get_ingredients在BasePizza中是一个类方法。

可以在抽象方法中做代码的实现?没错,Python与Java接口中的方法相反,你可以在抽象方法编写实现代码通过super()来调用它。(译注:在Java8中,接口也提供的默认方法,允许在接口中写方法的实现)

Python

import abc

class BasePizza(object):
__metaclass__ = abc.ABCMeta

default_ingredients = ['cheese']

@classmethod
@abc.abstractmethod
def get_ingredients(cls):
"""Returns the ingredient list."""
return cls.default_ingredients

class DietPizza(BasePizza):
def get_ingredients(self):
return ['egg'] + super(DietPizza, self).get_ingredients()

这个例子中,你构建的每个pizza都通过继承BasePizza的方式,你不得不覆盖get_ingredients方法,但是能够使用默认机制通过super()来获取ingredient列表。

㈡ 如何启动django服务器

Today',,don',startnow!

,

YoucandirectlyCtrl+c,turnitoffandrunpythonmanager.pyrunservertorestart.

.,.

Ifyourproblemissolved,pleaseadopt!

Ifnot,keepasking!

etimeTheotherday,(crawlerweb)onthecampusintranet,.,,soIwillexperimentwithit.

Withoutfurtherado,.

-.

Oncealltheabovestepsaredone,thereisonemoresteptodo,.Opensettings._HOSTS=[]toALLOWED_HOSTS=["*"].

.pyislocatedandenterthecommand:

python3manage.pyrunserver0.0.0.0:8000

:TheserverIPaddress:8000,you'redone!

Attention:

1)python3isnotspecific,.

2..

,afterclosingtheterminal,,nohupcommand,commandisthepython3manage.pyrunserver0.0.0.0:8000mentionedabove.

,andthewebneedsserverstorun,suchasthecommonlyusednginx,apache,uwsgi,etc.,(projectswrittenbydjango),,,

Python3.(PartI)Python3.(PartI)

1.CMDviewthepythonversion

2.CMDtoviewthepipversion

3,pipinstallsDjango,defaultDjango3

Upgradepip

4)CheckouttheDjangoversion

5.InstallMySQL

6.ChecktheMySQLversion

7.CreatetheDjangofolderandenter,

8)StartaDjangoproject

9.

10.

DjangoSourceCodeReading(1)ProjectbuildandstartTobehonest,Ihaven'tappreciatedDjangosofar.Inmymind,it'snotaveryelaboratedesign.Justa"maturescheme"stackepoffeatures..Nomatterhowmuchyoudon'tlikeit,butit'sneeded.Hopefully,oneday,,.(Endof)

,,onvenience.,.

django-,andthecommandlineisexeformat.

manage..

execute_from_command_line()-linearguments.Thenrunhisexecute().

Ifreloadisset,itwillbecheck_errorsbeforestarting.

check_errors()isaclosure,sotheaboveendswith(django.setup)().

.INSTALLED_APPS。GrabtheappfromSettings

.pyinourproject.Instead,anobjectislocated__init__.pyDjangoConf

guntilthe__getattr__value..__dict__(,because__getattr__hasalowerpriority)

Tofacilitatedebugging,let'swritearun.pydirectly.Nocommandlineapproach.

Createarun.

Debuggrabsetting_mole

()apps.populate(settings.INSTALLED_APPS)

Startlookingatapps.populate()

Let'sstartwiththisparagraph

.Itisloadedintotheself.app_configsdictionary

Then,calltheimport_models()andready().

Tofacilitatedebugging,let'srephrasethelastsentence

Thetypeofres,isCommanddjango.contrib.staticfiles.management.commands.runserver.

Thefocusisonthesecondsentence,let'sskiptotherun_from_argv()method,.

lnotgetthecorrectdirection.Infact,.

Therearetwocaseshere,ifitisreloadoverloading,theinner_run()willbeexecuteddirectly,.

WhenaDjangoprojectstarts,itactuallystartstwice,(manage.py),.

Onthefirstboot,theDJANGO_AUTORELOAD_.willenterrestart_with_reloader().

HeretheDJANGO_AUTORELOAD_ENVissettoTrueandthenrebooted.

Thesecondtime,youcanenterthestartuplogic.

_run().

.run(django_main_thread).

Let'slooknextatDjango'smainthreadinner_run().

WhenweseeWSGI,,endsthere.

,.(AsgiwasalsoproposedbyDjango,)

SowheredidthisWSGIcomefrom?Let'sgobackabit

...pyprofile.

Let'slookforthisget_wsgi_application().

Itcallssetup()againand,importantly,.

That'sWSGIAPPitself.

load_middleware()为构建中间件堆栈,这也是wsgiapp获取setting信息的唯一途径。导入settings.py,生成中间件堆栈。

如果看过我之前那篇fastapi源码的,应该对中间件堆栈不陌生。

app入口→中间件堆栈→路由→路由节点→endpoint

所以,wsgiapp就此构建完毕,服务器传入请求至app入口,即可经过中间件到达路由进行分发。

结语:以上就是首席CTO笔记为大家整理的关于如何启动django服务器的全部内容了,感谢您花时间阅读本站内容,希望对您有所帮助,更多关于如何启动django服务器的相关内容别忘了在本站进行查找喔。

㈢ python super()用法遇到TypeError: must be type, not classobj

第一关于

super().__init__()

这种不指定子类的继承方法是在Python 3中新加入的,因为你使用的是Python 2, 所以会报错。


第二,关于TypeError,请题主在定义两个父类的时候加入他们的父类(object),如下,再尝试一次,因为Python 2中的classobj不算做一种type,如果你用Python 3的话应该就没有这种问题了。

classBase1(object):
def__init__(self):
print"我是Base1"

classBase2(object):
def__init__(self):
print"我是Base2"


另外一点题外话,题主对Base这个子类采用了双重继承,即此子类同时拥有两个父类(Base1和Base2),这种继承方法有时容易造成混乱,即如果两个父类有相同的方法名或变量名时,无法确定应继承哪一个。因此在诸如Java等语言中多重继承是不允许的。Python允许多重继承,但在这种情况下为了避免混乱,Python会自动按顺序选择拥有所需方法名的第一个父类并执行,而忽略其他父类中的同名方法。比如你的

classBase(Base1,Base2):
def__init__(self):
super(Base,self).__init__()

中,使用

super(Base,self).__init__()

来呼叫Base父类的__init__方法,由于Base的父类有两个(Base1和Base2),并且两个都有__init__方法,所以Python默认只按顺序呼叫Base1的__init__方法而忽略Base2的。这样你运行

bbbb=Base()

时就只会显示

我是Base1

而不会显示“我是Base2”。如果你要使用super()方法并希望像使用

classBase(Base1,Base2):
def__init__(self):
Base1.__init__(self)
Base2.__init__(self)

时一样同时输出“我是Base1”和“我是Base2”的话,你需要修改代码使Base2为Base1的子类,Base为Base2的子类才可以。如下:

classBase1(object):
def__init__(self):
print"我是Base1"

classBase2(Base1):
def__init__(self):
super(Base2,self).__init__()
print"我是Base2"

classBase(Base2):
def__init__(self):
super(Base,self).__init__()

这样你运行

bbbb=Base()

时才会输出

我是Base1
我是Base2

㈣ 如何在Python中使用static,class,abstract方法

方法在Python中是如何工作的

方法就是一个函数,它作为一个类属性而存在,你可以用如下方式来声明、访问一个函数:

Python

1

2

3

4

5

6

7

8

>>> class Pizza(object):

... def __init__(self, size):

... self.size = size

... def get_size(self):

... return self.size

...

>>> Pizza.get_size

<unbound method Pizza.get_size>

Python在告诉你,属性_get_size是类Pizza的一个未绑定方法。这是什么意思呢?很快我们就会知道答案:

Python

1

2

3

4

>>> Pizza.get_size()

Traceback (most recent call last):

File "<stdin>", line 1, in <mole>

TypeError: unbound method get_size() must be called with Pizza instance as first argument (got nothing instead)

我们不能这么调用,因为它还没有绑定到Pizza类的任何实例上,它需要一个实例作为第一个参数传递进去(Python2必须是该类的实例,Python3中可以是任何东西),尝试一下:

Python

1

2

>>> Pizza.get_size(Pizza(42))

42

太棒了,现在用一个实例作为它的的第一个参数来调用,整个世界都清静了,如果我说这种调用方式还不是最方便的,你也会这么认为的;没错,现在每次调用这个方法的时候我们都不得不引用这个类,如果不知道哪个类是我们的对象,长期看来这种方式是行不通的。

那么Python为我们做了什么呢,它绑定了所有来自类_Pizza的方法以及该类的任何一个实例的方法。也就意味着现在属性get_size是Pizza的一个实例对象的绑定方法,这个方法的第一个参数就是该实例本身。

Python

1

2

3

4

>>> Pizza(42).get_size

<bound method Pizza.get_size of <__main__.Pizza object at 0x7f3138827910>>

>>> Pizza(42).get_size()

42

和我们预期的一样,现在不再需要提供任何参数给_get_size,因为它已经是绑定的,它的self参数会自动地设置给Pizza实例,下面代码是最好的证明:

Python

1

2

3

>>> m = Pizza(42).get_size

>>> m()

42

更有甚者,你都没必要使用持有Pizza对象的引用了,因为该方法已经绑定到了这个对象,所以这个方法对它自己来说是已经足够了。

也许,如果你想知道这个绑定的方法是绑定在哪个对象上,下面这种手段就能得知:

Python

1

2

3

4

5

6

7

>>> m = Pizza(42).get_size

>>> m.__self__

<__main__.Pizza object at 0x7f3138827910>

>>> # You could guess, look at this:

...

>>> m == m.__self__.get_size

True

显然,该对象仍然有一个引用存在,只要你愿意你还是可以把它找回来。

在Python3中,依附在类上的函数不再当作是未绑定的方法,而是把它当作一个简单地函数,如果有必要它会绑定到一个对象身上去,原则依然和Python2保持一致,但是模块更简洁:

Python

1

2

3

4

5

6

7

8

>>> class Pizza(object):

... def __init__(self, size):

... self.size = size

... def get_size(self):

... return self.size

...

>>> Pizza.get_size

<function Pizza.get_size at 0x7f307f984dd0>

静态方法

静态方法是一类特殊的方法,有时你可能需要写一个属于这个类的方法,但是这些代码完全不会使用到实例对象本身,例如:

Python

1

2

3

4

5

6

7

class Pizza(object):

@staticmethod

def mix_ingredients(x, y):

return x + y

def cook(self):

return self.mix_ingredients(self.cheese, self.vegetables)

这个例子中,如果把_mix_ingredients作为非静态方法同样可以运行,但是它要提供self参数,而这个参数在方法中根本不会被使用到。这里的@staticmethod装饰器可以给我们带来一些好处:

  • Python不再需要为Pizza对象实例初始化一个绑定方法,绑定方法同样是对象,但是创建他们需要成本,而静态方法就可以避免这些。

  • Python

    1

    2

    3

    4

    5

    6

  • >>> Pizza().cook is Pizza().cook

    False

    >>> Pizza().mix_ingredients is Pizza.mix_ingredients

    True

    >>> Pizza().mix_ingredients is Pizza().mix_ingredients

    True

  • 可读性更好的代码,看到@staticmethod我们就知道这个方法并不需要依赖对象本身的状态。

  • 可以在子类中被覆盖,如果是把mix_ingredients作为模块的顶层函数,那么继承自Pizza的子类就没法改变pizza的mix_ingredients了如果不覆盖cook的话。

  • 类方法

    话虽如此,什么是类方法呢?类方法不是绑定到对象上,而是绑定在类上的方法。

    Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

    13

    14

    15

  • >>> class Pizza(object):

    ... radius = 42

    ... @classmethod

    ... def get_radius(cls):

    ... return cls.radius

    ...

    >>>

    >>> Pizza.get_radius

    <bound method type.get_radius of <class '__main__.Pizza'>>

    >>> Pizza().get_radius

    <bound method type.get_radius of <class '__main__.Pizza'>>

    >>> Pizza.get_radius is Pizza().get_radius

    True

    >>> Pizza.get_radius()

    42

  • 无论你用哪种方式访问这个方法,它总是绑定到了这个类身上,它的第一个参数是这个类本身(记住:类也是对象)。

    什么时候使用这种方法呢?类方法通常在以下两种场景是非常有用的:

  • 工厂方法:它用于创建类的实例,例如一些预处理。如果使用@staticmethod代替,那我们不得不硬编码Pizza类名在函数中,这使得任何继承Pizza的类都不能使用我们这个工厂方法给它自己用。

  • Python

    1

    2

    3

    4

    5

    6

    7

  • class Pizza(object):

    def __init__(self, ingredients):

    self.ingredients = ingredients

    @classmethod

    def from_fridge(cls, fridge):

    return cls(fridge.get_cheese() + fridge.get_vegetables())

  • 调用静态类:如果你把一个静态方法拆分成多个静态方法,除非你使用类方法,否则你还是得硬编码类名。使用这种方式声明方法,Pizza类名明永远都不会在被直接引用,继承和方法覆盖都可以完美的工作。

  • Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

    13

    14

    15

  • class Pizza(object):

    def __init__(self, radius, height):

    self.radius = radius

    self.height = height

    @staticmethod

    def compute_area(radius):

    return math.pi * (radius ** 2)

    @classmethod

    def compute_volume(cls, height, radius):

    return height * cls.compute_area(radius)

    def get_volume(self):

    return self.compute_volume(self.height, self.radius)

  • 抽象方法

    抽象方法是定义在基类中的一种方法,它没有提供任何实现,类似于Java中接口(Interface)里面的方法。

    在Python中实现抽象方法最简单地方式是:

    Python

    1

    2

    3

  • class Pizza(object):

    def get_radius(self):

    raise NotImplementedError

  • 任何继承自_Pizza的类必须覆盖实现方法get_radius,否则会抛出异常。

    这种抽象方法的实现有它的弊端,如果你写一个类继承Pizza,但是忘记实现get_radius,异常只有在你真正使用的时候才会抛出来。

    Python

    1

    2

    3

    4

    5

    6

    7

  • >>> Pizza()

    <__main__.Pizza object at 0x7fb747353d90>

    >>> Pizza().get_radius()

    Traceback (most recent call last):

    File "<stdin>", line 1, in <mole>

    File "<stdin>", line 3, in get_radius

    NotImplementedError

  • 还有一种方式可以让错误更早的触发,使用Python提供的abc模块,对象被初始化之后就可以抛出异常:

    Python

    1

    2

    3

    4

    5

    6

    7

    8

  • import abc

    class BasePizza(object):

    __metaclass__= abc.ABCMeta

    @abc.abstractmethod

    def get_radius(self):

    """Method that should do something."""

  • 使用abc后,当你尝试初始化BasePizza或者任何子类的时候立马就会得到一个TypeError,而无需等到真正调用get_radius的时候才发现异常。

    Python

    1

    2

    3

    4

  • >>> BasePizza()

    Traceback (most recent call last):

    File "<stdin>", line 1, in <mole>

    TypeError: Can't instantiate abstract class BasePizza with abstract methods get_radius

  • 混合静态方法、类方法、抽象方法

    当你开始构建类和继承结构时,混合使用这些装饰器的时候到了,所以这里列出了一些技巧。

    记住,声明一个抽象的方法,不会固定方法的原型,这就意味着虽然你必须实现它,但是我可以用任何参数列表来实现:

    Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

    13

  • import abc

    class BasePizza(object):

    __metaclass__= abc.ABCMeta

    @abc.abstractmethod

    def get_ingredients(self):

    """Returns the ingredient list."""

    class Calzone(BasePizza):

    def get_ingredients(self, with_egg=False):

    egg = Egg() if with_egg else None

    return self.ingredients + egg

  • 这样是允许的,因为Calzone满足BasePizza对象所定义的接口需求。同样我们也可以用一个类方法或静态方法来实现:

    Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

    13

  • import abc

    class BasePizza(object):

    __metaclass__= abc.ABCMeta

    @abc.abstractmethod

    def get_ingredients(self):

    """Returns the ingredient list."""

    class DietPizza(BasePizza):

    @staticmethod

    def get_ingredients():

    return None

  • 这同样是正确的,因为它遵循抽象类BasePizza设定的契约。事实上get_ingredients方法并不需要知道返回结果是什么,结果是实现细节,不是契约条件。

    因此,你不能强制抽象方法的实现是一个常规方法、或者是类方法还是静态方法,也没什么可争论的。从Python3开始(在Python2中不能如你期待的运行,见issue5867),在abstractmethod方法上面使用@staticmethod和@classmethod装饰器成为可能。

    Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

  • import abc

    class BasePizza(object):

    __metaclass__= abc.ABCMeta

    ingredient = ['cheese']

    @classmethod

    @abc.abstractmethod

    def get_ingredients(cls):

    """Returns the ingredient list."""

    return cls.ingredients

  • 别误会了,如果你认为它会强制子类作为一个类方法来实现get_ingredients那你就错了,它仅仅表示你实现的get_ingredients在BasePizza中是一个类方法。

    可以在抽象方法中做代码的实现?没错,Python与Java接口中的方法相反,你可以在抽象方法编写实现代码通过super()来调用它。(译注:在Java8中,接口也提供的默认方法,允许在接口中写方法的实现)

    Python

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    12

    13

    14

    15

    16

  • import abc

    class BasePizza(object):

    __metaclass__= abc.ABCMeta

    default_ingredients = ['cheese']

    @classmethod

    @abc.abstractmethod

    def get_ingredients(cls):

    """Returns the ingredient list."""

    return cls.default_ingredients

    class DietPizza(BasePizza):

    def get_ingredients(self):

    return ['egg'] + super(DietPizza, self).get_ingredients()

  • 这个例子中,你构建的每个pizza都通过继承BasePizza的方式,你不得不覆盖get_ingredients方法,但是能够使用默认机制通过super()来获取ingredient列表。

    打赏支持我翻译更多好文章,谢谢!

阅读全文

与pythonbaseclass相关的资料

热点内容
中行app下载的文件在哪里 浏览:203
命令字描述 浏览:530
安卓手机上怎么找所有照片 浏览:505
成为程序员的bug 浏览:857
linux防火墙设置端口 浏览:453
阿里推推app在哪里下载 浏览:21
南昌app开发哪个好 浏览:527
ftp服务器默认什么策略 浏览:281
为什么安卓机用一两年会变卡 浏览:462
51单片机adc模块 浏览:487
centosjava环境变量 浏览:16
纵向加密连交换机网口不亮 浏览:188
十天学会php下载 浏览:507
制谱app哪里好 浏览:711
androidvolley封装 浏览:595
安卓什么软件可以自动配图 浏览:91
框架梁没加密的箍筋怎么办 浏览:678
搜题软件python 浏览:29
银行解压拿什么证件 浏览:566
linux的dns设置方法 浏览:421