python重写方法和重写特殊构造方法

目录
  • 继承
  • 多继承的优缺点
  • 子类的方法__init__()
  • 重写父类的方法
  • 将实例用作属性

继承

编写类时,并非总是要从空白开始,如果编写的类是另一个现成类的特殊版本,可使用继承,继承分为单继承和多继承。

一个类继承另一个类时,将自动获得另一个类的所有属性和方法,原有的类称为父类,而新类称为子类。子类继承了父类所有的属性和方法,同时还可以定义自己的属性和方法,这样一来就解决了类类与类之间代码冗余的问题

那么儿子怎么查看自己的父亲是谁呢?

如下所示:

class Parent_1:
    pass
class Parent_2:
    pass
class sub1(Parent_1):#单继承
    pass
#查看自己的父类
print(sub1.__bases__)
class sub2(Parent_1,Parent_2):#多继承
    pass
#查看自己的父类
print(sub2.__bases__)
(<class '__main__.Parent1'>,)
(<class '__main__.Parent1'>, <class '__main__.Parent2'>)

多继承的优缺点

优点:子类可以同时遗传多个父类的属性,最大限度地重用代码 缺点:违背伦理道德,一个儿子可以有多个爹,体现在程序中则为代码地可读性变差。

继承查找的顺序:

对象>子类>父类>父父类

举例:

class Fu():
    def f1(self):
        print('Fu.f1')
    def f2(self):
        print('Fu.f2')
        self.f1()#对象名.方法(),此时的self==objects
class son(Fu):
    def f1(self):
        print('son.f1')
objects=son()
objects.f2()

根据继承查找的顺序,对象>子类>父类>父父类,先在objects空间范围内查找f2,如果未找到,再去子类空间范围内查找,最后再去父类空间范围内查找。

Foo.f2
Bar.f1

子类的方法__init__()

在既有类的基础上编写新类的时候,通常要调用父类的方法__init__(),这将初始化再父类__init__()方法中定义的所有属性,从而让子类包含这些属性。

举例:

#定义一个父类Car,父类又名超类,名称super由此而来
class Car:
    def __init__(self,make,model,year):
        self.make=make
        self.model=model
        self.year=year
        self.odometer_reading=0
    def get_descriptive_name(self):
        long_name=f"{self.year} {self.make} {self.model}"
        return  long_name.title()
    def read_odometer(self):
        print(f"this car has {self.odometer_reading} miles on it ")
    def update_odometer(self,mileage):
        if mileage>=self.odometer_reading:
            self.odometer_reading=mileage
        else:
            print("you can't roll back an odometer!")
    def increment_odometer(self,miles):
        self.odometer_reading+=miles
#定义一个子类ELectricCar,创建子类时,父类必须包含在当前的文件,父类必须位于子类的前面
class ELectricCar(Car):#定义子类时,必须在圆括号内指定父类的名称
   #方法__init__()接受创建Car实例所需的信息
    def __init__(self,make,model,year):
        print("__init__()方法被调用")
        #让python调用Car类的方法__init__(),让子类创建的实例包含父类这个方法中定义的所有属性
        super().__init__(make,model,year)#super是一个特殊函数,使我们能够调用父类的方法
my_tesla=ELectricCar('tesla','model s',2019)
print(my_tesla.make)
print(my_tesla.year)
print(my_tesla.model)
print(my_tesla.get_descriptive_name())
__init__()方法被调用
tesla
2019
model s
2019 Tesla Model S

对于上述代码,我们只是想查看子类ELectricCar是否继承了父类Car所拥有的属性,但是子类本身,我们并没有给他设置自身属性和方法。

给子类定义属性和方法:

让一个类继承另一个类后,就可以添加区分子类和父类所需的新属性和新方法了。

下面来添加一个电动车特有的属性,以及描述该属性的方法:

依然选用上面的代码:

#定义一个父类Car,父类又名超类,名称super由此而来
class Car:
    def __init__(self,make,model,year):
        self.make=make
        self.model=model
        self.year=year
        self.odometer_reading=0
    def get_descriptive_name(self):
        long_name=f"{self.year} {self.make} {self.model}"
        return  long_name.title()
    def read_odometer(self):
        print(f"this car has {self.odometer_reading} miles on it ")
    def update_odometer(self,mileage):
        if mileage>=self.odometer_reading:
            self.odometer_reading=mileage
        else:
            print("you can't roll back an odometer!")
    def increment_odometer(self,miles):
        self.odometer_reading+=miles
#定义一个子类ELectricCar,创建子类时,父类必须包含在当前的文件,父类必须位于子类的前面
class ELectricCar(Car):#定义子类时,必须在圆括号内指定父类的名称
   #方法__init__()接受创建Car实例所需的信息
    def __init__(self,make,model,year):
        print("__init__()方法被调用")
        #让python调用Car类的方法__init__(),让子类创建的实例包含父类这个方法中定义的所有属性
        super().__init__(make,model,year)#super是一个特殊函数,使我们能够调用父类的方法
        self.battery_size=75
    def describle_battery(self):
        print(f"this car a {self.battery_size}-kwl battery")
my_tesla=ELectricCar('tesla','model s',2019)
print(my_tesla.get_descriptive_name())
my_tesla.describle_battery() 

这时,我们给子类添加了它的专有属性describle_battery_size:

#__init__()方法被调用
2019 Tesla Model S
this car a 75-kwl battery

下面我们主要对新添加的子类专有属性进行分析:

  def __init__(self,make,model,year):
        print("__init__()方法被调用")
        #让python调用Car类的方法__init__(),让子类创建的实例包含父类这个方法中定义的所有属性
        super().__init__(make,model,year)#super是一个特殊函数,使我们能够调用父类的方法
        self.battery_size=75
def describle_battery(self):#关于子类ELectricCar特有的描述
        print(f"this car a {self.battery_size}-kwl battery")

self.battery_size=75为子类特有属性,因此写在子类的__init__()方法后,根据子类ELectricCar创建的所有实例都将把包含该属性,但所有的Car实例都不包含它。

对于子类的特殊程度没有任何限制,模拟子类ELectricCar时,可根据所需的准确程度添加任意数量的属性和方法。

如果一个属性或方法是任何汽车都有的,而不是子类ELectricCar特有的,就将应将其加入到父类Car中,而不是加入到子类ELectricCar中,这样,使用父类Car类的人将获得相应的功能,而使用子类ELectricCar的人只能获得子类特有的属性。

重写父类的方法

对于父类的方法,只要他不符合子类模拟的实物的行为,都可以进行重写,为此。可在子类中定义一个与要重写的父类方法同名的方法,这样,python将不会考虑这个父类方法,而只关注你在子类中定义的相应方法。

举例:

假设父类Car有一个名为fill__gas__tank()的方法,他对于子类ELectricCar来说毫无意义,因此你可能想重写它,那该怎么重写呢?
可在子类中定义一个与要重写的父类方法同名的方法

class Car:
  ---snip:
    def fill_gas_tank(self):
        self.fill_gas_tank=90
        print(f"电瓶车邮箱尺寸的大小是{self.fill_gas_tank}")
class ELectricCar(Car):
    --snip:
    def fill_gas_tank(self):#与父类中该属性的方法名相同
        print("this car doesn't need a gas tank!")
my_tesla=ELectricCar('tesla','model s',2019)
print(my_tesla.get_descriptive_name())
my_tesla.fill_gas_tank()

将父类改写之后,输出的不符合子类ELectricCar的方法的相关行为是我们改写后的,如果不进行改写,那么则会输出不相关的属性行为。

#__init__()方法被调用
2019 Tesla Model S
this car doesn't need a gas tank!

将实例用作属性

使用代码模拟实物时,你可能会发现自己给类添加的细节越来越多:属性和方法清单以及文件都越来越长,在这种情况下,可能需要将类的一部分提取出来,作为一个单独的类,可以将大型类拆分成多个协同工作的小类。

例如,不断给子类ELectricCar添加细节时,我们可能发现其中包含很多专门针对汽车电池的属性和方法,在这种情况下我们可以将这些属性和方法提取出来,放在一个名为battery的类中,并将一个battery实例作为子类ELectricCar的属性:

举例:

class Car:
    --snip--
class Battery:#这里是重写一个类
    def __init__(self,battery_size=10):#该默认值可设定也可不设定
        self.battery_size=battery_size
    def describle_battery(self):
        print(f"this car has a {self.battery_size}-kwl battery")
class ELectricCar(Car):
    def __init__(self,make,model,year):
        super().__init__(make,model,year)
        self.battery=Battery()#在子类ELectricCar新添加了一个名为battery的属性
        #让python创建一个新的Battery实例,并将该实例赋给属性新创建的属性battery
my_tesla=ELectricCar('tesla','model s',2019)
#和上面描述电池容量是一样的方法
my_tesla.battery.describle_battery()#让python在实例my_tasla中查找属性battery,并对battery进行调用

下面我们再向Battery类中添加一个方法用来描述电瓶车的航行距离:

class Car:
   --snip--
class Battery:
   --snip--
    def get_range(self):
        if self.battery_size==75:
            range = 260
        elif self.battery_size==100:
            range = 315
        print(f"this car can go about {range} miles on a full charge")
class ELectricCar(Car):
   --snip--
my_tesla=ELectricCar('tesla','model s',2019)
my_tesla.battery.describle_battery()
my_tesla.battery.get_range()#在my_tesla实例中查找battery,self.battery=Battery(),对Battery进行调用
#__init__被调用
this car has a 75-kwl battery
this car can go about 260 miles on a full charge

到此这篇关于python重写方法和重写特殊构造方法的文章就介绍到这了,更多相关python重写内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Python基础类继承重写实现原理解析

    1.继承 继承是面向对象三大特性之一 - 通过继承我们可以使用一个类获取到其他类中的属性和方法 - 在定义类时,可以在类名后的括号中指定当前类的父类(超类.基类.super) - 子类(衍生类)可以直接继承父类中的所有的属性和方法 注意事项: a.在创建类时,如果省略了父类,则默认父类为object, object是所有类的父类,所有类都继承自object. b. isinstance()用来检查一个对象是否是一个类的实例 如果这个类是这个对象的父类,也会返回True 所有的对象都是object

  • Python基础知识方法重写+文件处理+异常处理

    目录 一.方法重写 二.运算符重载 三.文件处理 打开关闭文件 文件读写 四.异常处理 异常 抛出异常 最后 前言; Python基础知识+结构+数据类型 Python基础学习列表+元组+字典+集合 Python基础学习函数+模块+类 今天给大家分享的是第四篇,前面几篇我都放在上面了.小伙伴可以自己巩固复习 一.方法重写 print("-------------方法重写------------"); # coding=utf-8 class Parent: # 定义父类 def myM

  • Python3之字符串比较_重写cmp函数方式

    目录 Python3字符串比较_重写cmp函数 不多说,直接上代码 字符串比较 cmp op.eq 直接使用cmp,会出现错误 python3 比较字符串如下 Python3字符串比较_重写cmp函数 由于在C ++中习惯了使用CMP函数,所以在遇到字符串排序时,想当然的去使用sort(开始,结束,CMP)去对列表进行排序,但结果好像不行. 后来查阅网上资料,好像在python3中CMP函数已经被取代了. 故而只能另求他法了.下面是很简单的一个字符串日期提取及根据日期排序. 需求是这样的,由于从

  • Python反射和内置方法重写操作详解

    本文实例讲述了Python反射和内置方法重写操作.分享给大家供大家参考,具体如下: isinstance和issubclass isinstance(obj,cls)检查是否obj是否是类 cls 的对象,类似 type() class Foo(object): pass obj = Foo() isinstance(obj, Foo) issubclass(sub, super)检查sub类是否是 super 类的派生类 class Foo(object): pass class Bar(Fo

  • Python 继承,重写,super()调用父类方法操作示例

    本文实例讲述了Python 继承,重写,super()调用父类方法操作.分享给大家供大家参考,具体如下: demo.py(继承,重写,super): # 父类 class Dog: def bark(self): print("汪汪叫") # 子类 继承 class XiaoTianQuan(Dog): def fly(self): print("我会飞") # 可以重写父类中的同名方法 def bark(self): # super().父类方法名 调用父类中的方

  • Python类成员继承重写的实现

    类成员的继承和重写 成员继承:子类继承了父类除构造方法外的所有成员 方法重写:子类可以重新定义父类中的方法,这样就会覆盖父类中的方法,也称为重写 代码如下 class Person: def __init__(self,name,age): self.name = name self.__age = age def say_age(self): print('我的年龄:',self.__age) def say_introduce(self): print('我的名字是{0}'.format(s

  • 对python中类的继承与方法重写介绍

    1.单继承 父类也叫基类 子类也叫派生类 如下所示,继承的关系: 继承的书写格式: class 子类(父类): 方法 实例: class Animal: def eat(self): print("-----吃-------") def drink(self): print("-----喝--------") class Dog(Animal): def drak(self): print("汪汪叫") a=Animal() a.eat() 孙类

  • python重写方法和重写特殊构造方法

    目录 继承 多继承的优缺点 子类的方法__init__() 重写父类的方法 将实例用作属性 继承 编写类时,并非总是要从空白开始,如果编写的类是另一个现成类的特殊版本,可使用继承,继承分为单继承和多继承. 一个类继承另一个类时,将自动获得另一个类的所有属性和方法,原有的类称为父类,而新类称为子类.子类继承了父类所有的属性和方法,同时还可以定义自己的属性和方法,这样一来就解决了类类与类之间代码冗余的问题 那么儿子怎么查看自己的父亲是谁呢? 如下所示: class Parent_1: pass cl

  • Java方法覆盖重写实现原理解析

    这篇文章主要介绍了Java方法覆盖重写实现原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 方法覆盖重写注意事项: 1.必须保证方法名相同,返回值也相同 @Override:写在方法前面,用来检测方法的覆盖重写是否有效,这个注解不是必要的,就算不写,方法覆盖重写符合要求也是正确的 2.子类方法的返回值必须[小于等于]父类方法的返回值 3.子类方法的修饰符必须[大于等于]父类方法的修饰符 继承关系中,父子类构造方法的访问特点: 1.子类构造

  • js重写方法的简单实现

    如下所示: Function.prototype.method = function(name, fn) { this.prototype[name] = fn;//this指的应该是Function.prototype对象 return this;//返回此对象,这个有助于像jq那样的级联操作 }; Number.method("interge",function(){ });//Number是一个构造函数,所以拥有上面的method方法,所以最后返回了Number对象(构造函数),

  • 对django的User模型和四种扩展/重写方法小结

    User模型 User模型是这个框架的核心部分.他的完整的路径是在django.contrib.auth.models.User.以下对这个User对象做一个简单了解: 字段: 内置的User模型拥有以下的字段: username: 用户名.150个字符以内.可以包含数字和英文字符,以及_.@.+..和-字符.不能为空,且必须唯一! first_name:歪果仁的first_name,在30个字符以内.可以为空. last_name:歪果仁的last_name,在150个字符以内.可以为空. e

  • Java中方法的重写与成员变量的隐藏

    这篇文章讨论了Java面向对象概念中一个基本的概念–Field Hiding(隐藏成员变量) 在讨论这个问题之前,我们看一段特别特别简单的代码,请问一下方法的数据结果是什么? /** * @author Hollis 17/9/27. */ public class FieldOverriding { public static void main(String[] args) { Sub c1 = new Sub(); System.out.println(" c1.s : " +

  • 详解java中的深拷贝和浅拷贝(clone()方法的重写、使用序列化实现真正的深拷贝)

    1.序列化实现 public class CloneUtils { @SuppressWarnings("unchecked") public static <T extends Serializable> T clone(T object){ T cloneObj = null; try { ByteArrayOutputStream out = new ByteArrayOutputStream(); ObjectOutputStream obs = new Objec

  • 简单的理解java集合中的HashSet和HashTree几个重写方法

    Java中的set是无序的,但是是不可重复的 HashSet底层是哈希表,通过调用hashcode和equals方法实现去重 当我们HashSet里面存的是字符串时,就能默认去重了,因为String已经重写了hashcode和euqals方法 public static void main(String[] args) { HashSet<String> set = new HashSet(); set.add("java"); set.add("c")

  • IOS内存泄漏检查方法及重写MLeakFinder

    对于iOS开发来讲,内存泄漏的问题,已经是老生常谈的话题.在日常的面试中经常会提到这些问题.我们日常的开发过程中进行内存泄漏的检测,一般是使用instrument工具中的Leaks/Allocation来进行排查,网络上也有比较高效又好用的内存泄漏检测工具,MLeakFinder. MLeakFinder-原理 首先看UIViewController,当一个UIViewController被pop或dismiss的时候,这个VC包括在这个VC上的View,或者子View都会很快的被释放.所以我们

  • Java和JVM的重载识别,重写方法是怎样进行的

    目录 1.案例 2.重载与重写 3.JVM的静态绑定和动态绑定 4.调用指令的符号引用 5.总结与实践 可变长参数方法的重载造成的. 1.案例 void invoke(Object obj, Object... args) { ... } void invoke(String s, Object obj, Object... args) { ... } invoke(null, 1);    // 调用第二个invoke方法 invoke(null, 1, 2); // 调用第二个invoke方

  • json跨域调用python的方法详解

    本文实例讲述了json跨域调用python的方法.分享给大家供大家参考,具体如下: 客户端: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml">

随机推荐