类与类之间的关系

本章主要内容:

  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

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

 

posted @ 2019-01-25 17:53  会飞的草帽  阅读(252)  评论(0编辑  收藏  举报