property装饰器_继承

property装饰器

一:装饰器

装饰器是在不修改被装饰对象源代码以及调用方式的前提下为被装饰对象添加

# 新功能的可调用对象
# print(property)

property是一个装饰器,是用来绑定给对象的方法伪造成一个数据属性

二:案例

"""
成人的BMI数值:
过轻:低于18.5
正常:18.5-23.9
过重:24-27
肥胖:28-32
非常肥胖, 高于32
  体质指数(BMI)=体重(kg)÷身高^2(m)
  EX:70kg÷(1.75×1.75)=22.86
"""
# 案例1:
class People:
    def __init__(self, name, weight, height):
        self.name = name
        self.weight = weight
        self.height = height

    # 定义函数的原因1:
    # 1、从bmi的公式上看,bmi应该是触发功能计算得到的
    # 2、bmi是随着身高、体重的变化而动态变化的,不是一个固定的值
    #    说白了,每次都是需要临时计算得到的

    # 但是bmi听起来更像是一个数据属性,而非功能
    @property
    def bmi(self):
        return self.weight / (self.height ** 2)

obj1 = People(‘egon‘, 90, 1.50)
# print(obj1.bmi())

obj1.height = 1.60
# print(obj1.bmi())

print(obj1.bmi)

# 输出:
35.15624999999999
egon
# 案例2:
class People:
    def __init__(self, name):
        self.__name = name

    # @property
    def get_name(self):
        return self.__name

    def set_name(self, val):
        if type(val) is not str:
            print(‘必须传入str类型‘)
            return
        self.__name = val

    def del_name(self):
        print(‘不让删除‘)
        # def self.__name

    name11 = property(get_name, set_name, del_name)

obj1 = People(‘egon‘)
# print(obj1.get_name())
print(obj1.get_name())

obj1.set_name(‘xxq‘)
print(obj1.get_name())
obj1.del_name()

# 输出:
egon
xxq
不让删除
egon
# 案例三:
class People:
    def __init__(self, name):
        self.__name = name

    @property
    def name(self):  # obj1.name
        return self.__name

    @name.setter
    def name(self, val):  # obj1.name=‘EGON‘
        if type(val) is not str:
            print(‘必须传入str类型‘)
            return
        self.__name = val

    @name.deleter
    def name(self):  # del obj1.name
        print(‘不让删除‘)
        # del self.__name

obj1 = People(‘egon‘)
# 人正常的思维逻辑
print(obj1.name)  #
# obj1.name=18
# del obj1.name

# 输出:
egon


继承

一:什么是继承

I:继承是一种创建新类的方式,新建的类可称为子类或派生类,父类又可称为基类或超类,子类会遗传父类的属性

II:需要注意的是:python支持多继承

在Python中,新建的类可以继承一个或多个父类



class Parent1(object):
    x = 1111
class Parent2(object):
    pass
class Sub1(Parent1): # 单继承
    pass
class Sub2(Parent1,Parent2): # 多继承
    pass
print(Sub1.__bases__)       # (<class ‘__main__.Parent1‘>,)
print(Sub2.__bases__)       # (<class ‘__main__.Parent1‘>, <class ‘__main__.Parent2‘>)
print(Sub1.x)               # 1111

ps1: 在Python2中有经典类与新式类之分

新式类:继承了object类的子类,以及该子类的子类子子类。。。

经典:没有继承object类的子类,以及该子类的子类子子类。。。

ps2:在Python3中没有继承任何类,那么会默认继承object类,所以Python3中所有的类都是新式类



print(Parent1.__bases__)        # (<class ‘object‘>,)
print(Parent2.__bases__)        # (<class ‘object‘>,)

III:Python的多继承

#     优点:子类可以同时遗传多个父类的属性,最大限度地重用代码
#     缺点:
#         1、违背人的思维习惯:继承表达的是一种什么"是"什么的关系
#         2、代码可读性会变差
#         3、不建议使用多继承,有可能会引发可恶的菱形问题,扩展性变差,
#         如果真的涉及到一个子类不可避免地要重用多个父类的属性,应该使用Mixins

二:为何要用继承

用来解决类与类之间代码冗余问题

三:如何实现继承

示范1:类与类之间存在冗余问题

class Student:
    school=‘OLDBOY‘

    def __init__(self, name, age, sex):
        self.name = name
        self.age = age
        self.sex = sex

    def choose_course(self):
        print(‘学生%s 正在选课‘ % self.name)

class Teacher:
    school=‘OLDBOY‘

    def __init__(self, name, age, sex, salary, level):
        self.name = name
        self.age = age
        self.sex = sex
        self.salary = salary
        self.level = level

    def score(self):
        print(‘老师 %s 正在给学生打分‘ % self.name)

示范2:基于继承解决类与类之间的冗余问题

class OldboyPeople:
    school = ‘OldBoy‘

    def __init__(self, name, age, sex):
        self.name = name
        self.age = age
        self.sex = sex

class Student(OldboyPeople):
    def choose_course(self):
        print(‘学生%s 正在选课‘ % self.name)

stu_obj = Student(‘lili‘, 18, ‘female‘)
# print(stu_obj.__dict__)     # {‘name‘: ‘lili‘, ‘age‘: 18, ‘sex‘: ‘female‘}
# print(stu_obj.school)         # OldBoy
# stu_obj.choose_course()       # 学生lili 正在选课

class Teacher(OldboyPeople):
    #           老师的空对象,‘egon‘,18,‘male‘,3000,10
    def __init__(self, name, age, sex, salary, level):
        # 指名道姓地跟父类OldboyPeople去要__init__
        OldboyPeople.__init__(self, name, age, sex)
        self.salary = salary
        self.level = level

    def score(self):
        print(‘老师 %s 正在给学生打分‘ % self.name)

tea_obj = Teacher(‘egon‘, 18, ‘male‘, 3000, 10)
# print(tea_obj.__dict__)     # {‘name‘: ‘egon‘, ‘age‘: 18, ‘sex‘: ‘male‘, ‘salary‘: 3000, ‘level‘: 10}
# print(tea_obj.school)       # OldBoy

tea_obj.score()               # 老师 egon 正在给学生打分
 

单继承背景下的属性查找

示范1:

class Foo:
    def f1(self):
        print(‘Foo.f1‘)

    def f2(self):
        print(‘Foo.f2‘)
        self.f1()  # obj.f1()

class Bar(Foo):
    def f1(self):
        print(‘Bar.f1‘)

obj = Bar()
obj.f2()

# 预料的结果
# Foo.f2
# Foo.f1

# 实际的结果
# Foo.f2
# Bar.f1
 

示范2:

class Foo:
    def f1(self):
        print(‘Foo.f1‘)

    def f2(self):
        print(‘Foo.f2‘)
        Foo.f1(self)  # 调用当前类中的f1

class Bar(Foo):
    def f1(self):
        print(‘Bar.f1‘)

obj = Bar()
obj.f2()

# 输出:
# Foo.f2
# Foo.f1
 

示范3:

class Foo:
    def __f1(self):  # _Foo__f1
        print(‘Foo.f1‘)

    def f2(self):
        print(‘Foo.f2‘)
        self.__f1()  # self._Foo__f1,# 调用当前类中的f1

class Bar(Foo):
    def __f1(self):  # _Bar__f1
        print(‘Bar.f1‘)

obj = Bar()
obj.f2()

# Foo.f2
# Foo.f1
 

多继承带来的菱形问题

一:菱形问题

  大多数面向对象语言都不支持多继承,而在Python中,一个子类是可以同时继承多个父类的,这固然可以带来一个子类可以对多个不同父类加以重用的好处,但也有可能引发著名的 Diamond problem菱形问题(或称钻石问题,有时候也被称为“死亡钻石”),菱形其实就是对下面这种继承结构的形象比喻。

class A(object):
    def test(self):
        print(‘from A‘)
    pass

class B(A):
    def test(self):
        print(‘from B‘)
    pass

class C(A):
    def test(self):
        print(‘from C‘)
    pass

class D(C, B):
    # def test(self):
    #     print(‘from D‘)
    pass

# print(D.mro())  # 类D以及类D的对象访问属性都是参照该类的mro列表
# 输出:[<class ‘__main__.D‘>, <class ‘__main__.C‘>, <class ‘__main__.B‘>, <class ‘__main__.A‘>, <class ‘object‘>]

obj = D()
obj.test()        # from C

print(D.test)       # <function C.test at 0x039D84A8>

print(C.mro())  # 类C以及类C的对象访问属性都是参照该类的mro列表
# 输出:[<class ‘__main__.C‘>, <class ‘__main__.A‘>, <class ‘object‘>]

c = C()
c.test()        # from C

总结:类相关的属性查找(类名.属性,该类的对象.属性),都是参照该类的mro


python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。 而这个MRO列表的构造是通过一个C3线性化算法来实现的。我们不去深究这个算法的数学原理,它实际上就是合并所有父类的MRO列表并遵循如下三条准则:

1.子类会先于父类被检查
2.多个父类会根据它们在列表中的顺序被检查
3.如果对下一个类存在两个合法的选择,选择第一个父类

二:如果多继承是非菱形继承,经典类与新式的属性查找顺序一样:

都是一个分支一个分支地找下去,然后最后找object

class E:
    # def test(self):
    #     print(‘from E‘)
    pass

class F:
    def test(self):
        print(‘from F‘)

class B(E):
    # def test(self):
    #     print(‘from B‘)
    pass

class C(F):
    # def test(self):
    #     print(‘from C‘)
    pass

class D:
    def test(self):
        print(‘from D‘)

class A(B, C, D):
    # def test(self):
    #     print(‘from A‘)
    pass

# 新式类
# print(A.mro()) # A->B->E->C->F->D->object

obj = A()
obj.test()  # 结果为:from F

三:如果多继承是菱形继承,经典类与新式类的属性查找顺序不一样:

经典类:深度优先,会在检索第一条分支的时候就直接一条道走到黑,即会检索大脑袋(共同的父类)

新式类:广度优先,会在检索最后一条分支的时候检索大脑袋

class G:  # 在python2中,未继承object的类及其子类,都是经典类
    # def test(self):
    #     print(‘from G‘)
    pass

class E(G):
    # def test(self):
    #     print(‘from E‘)
    pass

class F(G):
    def test(self):
        print(‘from F‘)

class B(E):
    # def test(self):
    #     print(‘from B‘)
    pass

class C(F):
    def test(self):
        print(‘from C‘)

class D(G):
    def test(self):
        print(‘from D‘)

class A(B, C, D):
    # def test(self):
    #     print(‘from A‘)
    pass

# 新式类
# print(A.mro()) # A->B->E->C->F->D->G->object

# 经典类:A->B->E->G->C->F->D
obj = A()
obj.test()  #

四:总结:

多继承到底要不用???

要用,但是规避几点问题

1.继承结构尽量不要过于复杂

2.推荐使用mixins机制:在多继承的背景下满足继承的什么"是"什么的关系

 

原文地址:https://www.cnblogs.com/2722127842qq-123/p/12677703.html

时间: 2024-10-14 03:44:29

property装饰器_继承的相关文章

面向对象之封装 及@property装饰器使用

目录 封装 1.封装的定义 2.封装的目的: 3.封装的三种方式 4.封装的优点 5.访问限制(封装) @property 装饰器 封装 1.封装的定义 ? 将复杂的丑陋的,隐私的细节隐藏到内部,对外提供简单的使用接口, 对外隐藏内部实现细节,并提供访问的接口; 2.封装的目的: 封装的目的: 面向对象的核心是对象二字,精髓在于整合,封装的目的其实就是把一堆数据属性和方法属性整合到对象中,我们可以把对象比喻成一个容器,其实就是为了把数据存入一个容器中.存的目的就是为了取的,那封装到对象中的好处就

python面向对象:组合、封装、property装饰器、多态

一.组合二.封装三.property装饰器四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 3. 如何用组合 ''' # class Foo: # aaa=1111 # def __init__(self,x,y): # self.x=x # self.y=y # # def func1(self): # print('Foo内的功能') # # # class Bar: # bbb=2222

访问可见性问题和@property装饰器

1.访问可见性问题 在Python中,属性和方法的访问权限只有两种,也就是公开的和私有的,如果希望属性是私有的,在给属性命名时可以用两个下划线作为开头.如下所示: # 私有变量,变量名前面加"__" # 如果非要使用私有变量,那么可以使用dir(class())去查看它真正的名字. # 私有变量/函数,在类内部可以直接调用. # 如果你想体现一个变量/函数特别重要你可以使用"_" 2.@property装饰器 之前我们讨论过Python中属性和方法访问权限的问题,我

一段小代码说明@property装饰器的用法

#coding:utf-8 """ 一段小代码说明@property装饰器的用法.__name是私有变量 外部不能通过foo.__name访问,但可以通过foo._Foo__name访问. 如果想通过点运算符设置和访问实例属性怎么办,@property派上用途了, 先将一个方法头上加上@property,这个方法就变成实例属性了,再加上这个@get_name.setter 装饰器(@属性名.setter)就可以通过赋值来设置属性. python北京周末培训班 https://

python staticmethod,classmethod方法的使用和区别以及property装饰器的作用

class Kls(object): def __init__(self, data): self.data = data def printd(self): print(self.data) @staticmethod def smethod(*arg): print('Static:', arg) @classmethod def cmethod(*arg): print('Class:', arg) >>> ik = Kls(23) >>> ik.printd()

python @property装饰器

@property装饰器 @property装饰器就是负责把一个方法变成属性调用把一个getter方法变成属性,只需要加上@property就可以了,此时,@property本身又创建了另一个装饰器@score.setter,负责把一个setter方法变成属性赋值class Student(object): @property def score(self): return self._score @score.setter def score(self, value): if not isin

3.1.8 property装饰器

在类的方法上加上一行@property 装饰器,会使得用户调用该函数属性时,就像调用数据属性一样,不需要加上() 比如想获取一些名词,再加上括号,容易使调用者忘记.动词才加()调用方法. 如下示例: '''BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属性,更便于理解) 成人的BMI数值: 过轻:低于18.5 正常:18.5-23.9 过重:24-27 肥胖:28-32 非常肥胖, 高于32 体质指数(BMI)=体重(kg)÷身高^2(m) EX:70

面向对象——property装饰器

property装饰器 property装饰器的作用,其实就是将将函数属性伪装成为属性的的装饰器 class People: def __init__(self,name,weight,height): self.name = name self.weight =weight self.height=height @property def bmi(self): return self.weight/(self.height**2) msj = People('msj',82,1.84) pri

Python中,关于@property装饰器

1.为什么使用@property装饰器?br/>在类中,当我么不想在外界直接调用到类的属性,或者不想展示属性的真实内容时,可以用到@property.它规定了我们直接用 对象名.属性名 获取对象属性时并不会直接取得对象的属性,而是通过调用@property装饰过的属性函数来给调用者反馈. 2.我们为什么不使用特定的方法来进行上面的操作?原因是因为太繁琐.例: class Person: def __init__(self, username, password) -> None: self.u