• 类与类之间的关系


    本章主要内容:

      1.依赖关系

      2.关联关系,组合关系,聚合关系

      3.继承关系,self 到底是什么鬼?

      4.类中的特殊成员

    一 类与类之间的依赖关系

      大千世界,万物之间皆有规则和规律,我们的类和对象 是对大千世界中的事物进行归类,那事物之间存在着相对应的关系,类与类之间也是同样如此,在面向对象的世界中,类与类中存在以下关系:

      1.依赖关系

      2.关联关系

      3.组合关系

      4.聚合关系

      5.继承关系

      6.实现关系

    由于 python  是一门 弱类型编程语言,并且所有的对象之间都是多态的关系,也即是说,所有的东西都可以当做对象来使用,所以我们在写代码的时候很容易形成以上关系,首先,我们先看第一种,也就是这些关系中紧密度最低的一个,依赖关系

      首先,我们设计一个场景,还是最初的那个例子,要把大象装冰箱,注意,在这个场景中,其实是存在两种事物的,一个是大象,一个是冰箱,大象是整个事件的操作者,冰箱负责被大象操作

    首先,写出两个类,一个是大象类,一个是冰箱类

    class Elphant:
    
        def __init__(self,name):
            self.name = name
        def open(self):
            pass
        def close(self):
            pass
    class Refrigerator:
        def open_door(self):
            print("冰箱门被打开")
        def close_door(self):
            print("冰箱门被关上")

      冰箱的功能非常简单,主要会开门关门就可以了,但是大象就没那么简单了,想想,大象开门和关门的时候是不是先找到这个冰箱,然后再进行开门,装自己,关门,注意,开门和关门的都是同一个冰箱,并且,大象有换冰箱的权利,想进哪个冰箱都可以,这时 大象类和冰箱类 的关系就没那么紧密了,因为大象可以指定任何一个冰箱. (我中有你,你中没我)

    class Elphant:
        def __init__(self,name):
            self.name = name
        def open(self,ref):
            print("大象要开门了,默念三声,开.....")
            ref.open_door()
        def take(self):
            print("大象自己钻进去")
        def close(self,ref):
            print("大象默念关门...")
            ref.close_door()
    class Refrigerator:
        def open_door(self):
            print("冰箱门被打开")
        def close_door(self):
            print("冰箱门被关上")
    # 造冰箱
    r = Refrigerator()
    # 造大象
    el = Elphant("神奇的大象")
    el.open(r)
    el.take()
    el.close(r)

      此时,我们说,大象和冰箱之间就是依赖关系,我用你,但是你不属于我,这种关系是最弱的,比如,公司和雇员之间,对于正式员工,肯定要签劳动合同,还得小心伺候着,但是如果是兼职,那就无所谓了,需要了你就来,不需要你的生活,你就可以拜拜了,这里的兼职(零时工)就属于依赖关系,我用你,但是你不属于我

    二 关联关系,组合关系,聚合关系

      其实 这三个在代码上写法是一样的,但是从含义上是不一样的

      1.关联关系:两种事物必须是相互关联的,但是在某些特殊情况下是可以更改和更换 的

      2.聚合关系:属于关联关系中的一种特例,侧重点是 xxx 和xxx 聚合成 xxx,各自有各自的声明周期,比如电脑,电脑里有CPU,硬盘,内存等,电脑挂了,内存,硬盘都是好的,还是完整的个体

      3.组合关系,属于关联关系中的一种特例,写法差不多,组合关系比聚合关系还要紧密,比如人的大脑,心脏各个器官,这些器官组合成一个人,这时,人如果挂了,其他的东西也跟着挂了

    首先我们来看关联关系,这个最简单,也是最常用的一种关系,比如,大家都有男女朋友,男人关联着女人,女人关联着男人,这种关系可以是相互的,也可以是单方面的

    class Boy:
        def __init__(self,name,girlfriend = None):
            self.name = name
            self.girlfriend = girlfriend
    
        def eat(self):
            if self.girlfriend:
                print("%s和他的女朋友%s一起吃火锅" % (self.name,self.girlfriend.name))
            else:
                print("那个人好奇怪啊,你看他像一条狗啊")
    class Girl:
    
        def __init__(self,name):
            self.name = name
    # b = Boy("至尊宝")
    # b.eat()    
    # 那个人好奇怪啊,你看他像一条狗啊
    # # 出生自带女朋友 # g = Girl("小狐狸") # b = Boy("至尊宝",g) # b.eat()
    # 至尊宝和他的女朋友小狐狸一起吃火锅
    # # 突然有一天,小狐狸不在了 # b = Boy("至尊宝") # g = Girl("小狐狸") # b.girlfriend = None # 此时没有女朋友了 # b.eat()
    # 那个人好奇怪啊,你看他像一条狗啊
    # 直到有一天,碰到了紫霞仙子 g = Girl("紫霞仙子") b = Boy("至尊宝") b.girlfriend = g b.eat()
    # 至尊宝和他的女朋友紫霞仙子一起吃火锅

      注意,此时Boy 和 Girl 两个类之间就是 关联关系,两个类的对象紧密联系着,其中一个没有了,另外一个就孤单的不得了,关联关系,其实就是,我需要你,你也属于我,这就是关联关系,像这样的关系有很多很多,

    比如,学校和老师的关系

      老师必然属于一个学校,换句话说,每个老师肯定有一个指定的工作机构,就是学校,那老师的属性中必然关联着学校

    class School:
        def __init__(self,name,address):
            self.name = name
            self.address = address
    
    class Teacher:
        def __init__(self,name,school=None):
            self.name = name
            self.school = school
    s1 = School("花果学院","花果山水帘洞")
    s2 = School("佛学院","西方极乐世界")
    s3 = School("仙学院","88重天")
    
    t1 = Teacher("齐天大圣",s1)
    t2 = Teacher("紫霞仙子",s1)
    t3 = Teacher("如来佛祖",s2)
    t4 = Teacher("太上老君",s3)
    
    # 找到 太上老君 所在的校区地址
    print(t4.school.address) # 88重天

      想一想啊,这样的关系如果反过来,一个老师可以选一个学校任职,那反过来,一个学校有多少老师呢?

    一堆吧?这样的关系如何来描述呢??

    class School:
        def __init__(self,name,address):
            self.name = name
            self.address = address
            self.lst = []    # 每个学校都应该有一个装老师的列表
    
        def add_teacher(self,teacher):
            self.lst.append(teacher)
    
    class Teacher:
        def __init__(self,name,school=None):
            self.name = name
            self.school = school
    
    s1 = School("花果学院","花果山水帘洞")
    s2 = School("佛学院","西方极乐世界")
    s3 = School("仙学院","88重天")
    
    t1 = Teacher("齐天大圣",s1)
    t2 = Teacher("紫霞仙子",s1)
    t3 = Teacher("如来佛祖",s2)
    t4 = Teacher("太上老君",s3)
    
    s1.add_teacher(t1)
    s1.add_teacher(t2)
    s1.add_teacher(t3)
    s1.add_teacher(t4)
    
    for el in s1.lst:
        print(el.name)

    # 齐天大圣
    # 紫霞仙子
    # 如来佛祖
    # 太上老君

      好了,这就是关联关系,当我们在逻辑上出现了,我需要你,你还得属于我,这种逻辑,就是关联关系,那注意,这种关系的紧密程度比上面的依赖关系要紧密的多,为什么呢?想想吧

      至于组合关系和聚合关系,其实代码上的差别不大,都是把另一个类的对象作为这个类的属性来传递和保存,只是在含义上会有些许的不同而已

    三 继承关系

      在面向对象的世界中存在着继承关系,我们现实中也存在着这样的关系,我们说过,X是一种y,那x就可以继承y,这时理解层面上的,如果上升到代码层面,我们可以这样认为,子类在不影响父类的程序运行的基础上对父类进行的扩充和扩展,这里,我们可以把父类被称为超类或者基类,子类被称为派生类

      首先,类名和对象默认是可以作为字典的 key的

    class Foo:
        def __init__(self):
            pass
     
        def method(self):
            pass
    
    print(hash(Foo))
    print(hash(Foo()))

    # -2145808608
    # 1674773

      既然可以 hash,那就是说字典的 key 可以是对象或者类

    class Foo:
        def __init__(self):
            pass
    
        def method(self):
            pass
    dic = {}
    dic[Foo] = 123
    dic[Foo()] = 456
    print(dic)
    # {<class '__main__.Foo'>: 123, <__main__.Foo object at 0x01A0E110>: 456}

      虽然看起来有点诡异,但是是可以用的

      接下来,我们来继续研究继承上的相关内容,在本章中主要研究一下 self ,记住,不管方法之间如何进行调用,类与类之间是何关系,默认的self 都是访问这个方法的对象

    class Base:
    
        def __init__(self,num):
            self.num = num
        def func1(self):
            print(self.num)
    class Foo(Base):
        pass
    
    obj = Foo(123)
    obj.func1()   
    
    
     
    # 123   运行的是 Base 中的 func1

      继续:

    class Base:
    
        def __init__(self,num):
            self.num = num
        def func1(self):
            print(self.num)
    class Foo(Base):
        def func1(self):
            print("Foo.func1",self.num)
    
    obj = Foo(123)
    obj.func1()
    
    
    # 运行的是 Foo中的 func1
    # Foo.func1 123

      再来:

    class Base:
        def __init__(self,num):
            self.num = num
        def func1(self):
            print(self.num)
            self.func2()
        def func2(self):
            print("Base.func2")
    
    class Foo(Base):
        def func2(self):
            print("Foo.func2")
    
    obj = Foo(123)
    obj.func1()
    
    # 123          func1 是 Base 中的
    # Foo.func2    func2 是子类的

      总结: self 在访问方法的顺序,永远先找自己的,自己的找不到再找父类的

      接下来,再看看:

    class Base:
        def __init__(self,num):
            self.num = num
        def func1(self):
            print(self.num)
            self.func2()
        def func2(self):
            print(111,self.num)
    
    class Foo(Base):
        def func2(self):
            print(222,self.num)
    
    lst = [Base(1),Base(2),Foo(3)]
    for obj in lst:
        obj.func2()
    
    # 111 1
    # 111 2
    # 222 3

      再来,还不够绕..

    class Base:
        def __init__(self,num):
            self.num = num
        def func1(self):
            print(self.num)
            self.func2()
        def func2(self):
            print(111,self.num)
    
    class Foo(Base):
        def func2(self):
            print(222,self.num)
    
    lst = [Base(1),Base(2),Foo(3)]
    for obj in lst:
        obj.func1()

    # 看明白了嘛?????
    # 1 # 111 1 # 2 # 111 2 # 3 # 222 3

       结论: self 就是你访问方法的那个对象,先找自己,然后再找父类的 

    四 类中的特殊成员

      什么是特殊成员呢? __init__()就是 一个特殊的成员,说白了,带双下划线的那一坨,这些方法在特殊的场景的时候会被自动的执行,比如:

      1.类名() 会自动执行__init__()

      2.对象()会自动执行__call__()

      3.对象[key]会自动执行__getitem__()

      4.对象[key] = value 会自动执行__getitem__()

      5.del 对象[key]会自动执行__delitem__()

      6.对象 + 对象 会自动执行__add__()

      7.with 对象 as 变量 会自动执行__enter__和__exit__

      8.打印对象的时候 会自动执行__str__

      9.干掉可哈希 __hash__==None 对象就不可哈希了

    创建对象的真正步骤:

      首先,在执行类名()的时候,系统会自动先执行__new__() 来开辟内存,此时新开辟出来的内存区域是空的,紧随其后,系统自动调用__init__()来完成对象的初始化工作,按照时间轴来算

      1.加载类

      2.开辟内存(__new__)

      3.初始化(__init__)

      4.使用对象干 xxxxxx

      类似的操作还有很多很多,我们不需要完全刻意的去把所有的特殊成员全都记住,实战中也用不到那么多,用到查就是了.

  • 相关阅读:
    Unity3D实践系列08, MonoBehaviour类的各种触发事件
    Unity3D实践系列07,组件的启用或禁用开关,物体的的可见或不可见开关,以及相应事件
    Unity3D实践系列06,球体撞击物体游戏
    Linux tee的花式用法和pee
    bash内置命令mapfile:读取文件内容到数组
    Perl正则表达式引用
    Perl文件句柄引用
    透明代理、正向代理、反向代理的区别说明
    Perl回调函数和闭包
    一文搞懂:词法作用域、动态作用域、回调函数、闭包
  • 原文地址:https://www.cnblogs.com/wenqi2121/p/10320958.html
Copyright © 2020-2023  润新知