类和对象全解2(python)

  1. 以后定义属性的时候,不要让外界直接用,而是需要定义一个方法。示例如下:
class Dog:
    def set_age(self, new_age):
        if new_age > 0 and new_age <= 100:
            self.age = new_age
        else:
            self.age = 0

    def get_age(self):
        return self.age

dog = Dog()
#dog.age = -10
#dog.name = "小白"
#print(dog.age)

dog.set_age(-10)
age = dog.get_age()
print(age)

#dog.get_age()
#dog.get_name()
  1. 私有方法
class Msg:

    #私有方法
    def __send_msg(self):
        print("-----正在发送短信-----")

    #公有方法
    def send_msg(self, new_money):
        if new_money > 10000:
            self.__send_msg()
        else:
            print("余额不足,请先充值,再发送短信")

msg = Msg()
msg.send_msg(10000000)
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
-----正在发送短信-----

Process finished with exit code 0

这种开发的思想,比如开发一个发送短信的功能。不能直接让你调用这个私有的函数,我得先验证你的条件,如果满足就调用,不满足就不调用。换句话说,在实际开发中,如果一个函数前面有两个下划线,就意味着这个方法一般比较重要,不想让外面直接用。而是需要先调用一个方法也就是send_msg(),相当于先去验证。如果验证通过了,然后在方法里面self.__send_msg()再调用真正核心的方法。

  1. __del __ 方法
class Dog:
    def __del__(self):
        print("-----英雄over-----")


dog1 = Dog()
dog2 = dog1

del dog1
print("=================")
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
=================
-----英雄over-----

Process finished with exit code 0

此处指的注意的是,最下面的先打印,而del函数里面的之后打印,这是为什么呢?此外重新改一下就会发现结果会发生变化

class Dog:
    def __del__(self):
        print("-----英雄over-----")


dog1 = Dog()
dog2 = dog1

del dog1
del dog2
print("=================")
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
-----英雄over-----
=================

Process finished with exit code 0

这是因为,如果使用程序引用计数来描述这个步骤的话,当执行"dog1 = Dog()"之后,计数为1,当执行"dog2 = dog1"之后,计数为2。而当执行"del dog1"之后,计数减1,当执行"del dog2"之后,计数归零。此时,就要调用"__del __"函数,执行上面的print,之后再执行下面的print
而第一个程序中,在执行完下面的print之后,程序准备结束并把内存信息归零的时候,此时执行"__del __"函数,所以会出现先print下面的信息,再print函数里面的信息的情况。
以上的知识点可以用于比如说英雄联盟里面,在英雄被击杀之后,需要调用一个类似于喊叫,倒地的程序。

  1. 测量一个对象引用计数的方法
>>> import sys
>>> class T:
    pass

>>> t = T()
>>> sys.getrefcount(t)
2
#此处我们发现,数量一定比实际的多一个
>>> tt = t
>>> sys.getrefcount(tt)
3
>>> del tt
>>> sys.getrefcount(t)
2
>>> del t
>>> sys.getrefcount(t)
Traceback (most recent call last):
  File "<pyshell#13>", line 1, in <module>
    sys.getrefcount(t)
NameError: name 't' is not defined
>>> 
  1. 继承
#继承只需要在子类的class里面插入父类的名称就可以使用父类的函数功能了
#但是需要注意的是,dog只能调用Dog()和Animal里面的功能,不能调用Cat()里面的功能,反之亦然
class Animal:
    def eat(self):
        print("---eat---")

    def drink(self):
        print("---drink---")

    def sleep(self):
        print("---sleep---")

    def run(self):
        print("---run---")

class Dog(Animal):
    def bark(self):
        print("---shout---")

class Cat(Animal):
    def catch(self):
        print("--catching---")

dog = Dog()
dog.eat()
dog.bark()

tom = Cat()
tom.eat()
tom.catch()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
---eat---
---shout---
---eat---
--catching---

Process finished with exit code 0

还有一个神奇的功能,如果想要定义一个啸天犬类能够继承Dog的功能,代码如下

class Animal:
    def eat(self):
        print("---eat---")

    def drink(self):
        print("---drink---")

    def sleep(self):
        print("---sleep---")

    def run(self):
        print("---run---")

class Dog(Animal):
    def bark(self):
        print("---shout---")

class Xiaotianquan(Dog):
    def fly(self):
        print("---fly---")

xx = Xiaotianquan()
xx.eat()
xx.bark()
xx.fly()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
---eat---
---shout---
---fly---

Process finished with exit code 0

实现一层继承一层的功能,等于是将一个大类细分成很多个小类

  1. 重写
    如果说定义的最小子类Xiaotianquan()的bark()函数实现的跟Dog()不一样,那么我们就可以重写一个函数实现一个新的功能。
    如下的代码就是重写一个bark()的功能:
class Animal:
    def eat(self):
        print("---eat---")

    def drink(self):
        print("---drink---")

    def sleep(self):
        print("---sleep---")

    def run(self):
        print("---run---")

class Dog(Animal):
    def bark(self):
        print("---shout---")

class Xiaotianquan(Dog):
    def fly(self):
        print("---fly---")
    
    #在此处等于是重写bark()的功能,让它不再使用Dog()里面的函数
    def bark(self):
        print("---LOL---")

xx = Xiaotianquan()
xx.eat()
xx.bark()
xx.fly()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
---eat---
---LOL---
---fly---

Process finished with exit code 0

如果想要调用被重写的方法,有两种操作

class Animal:
    def eat(self):
        print("---eat---")

    def drink(self):
        print("---drink---")

    def sleep(self):
        print("---sleep---")

    def run(self):
        print("---run---")

    def bark(self):
        print("---测试一下---")

class Dog(Animal):
    def bark(self):
        print("---shout---")

class Xiaotianquan(Dog):
    def fly(self):
        print("---fly---")

    #在此处等于是重写bark()的功能,让它不再使用Dog()里面的函数
    def bark(self):
        print("---LOL---")

        #第一种调用被重写的父类的方法
        #Dog.bark(self)

        #第二种方法
        super().bark()

xx = Xiaotianquan()
xx.eat()
xx.bark()
xx.fly()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
---eat---
---LOL---
---shout---
---fly---

Process finished with exit code 0

经过测试发现,这两种调用的方法都是只能调用上一层的同名函数bark()的功能,我在Animal这个class里面也谢了一个bark()的函数作为测试,发现并没有调用。

  1. 私有方法、私有属性在继承中的表现
    第一种:
class A:
    def __init__(self):
        self.num1 = 100
        self.__num2 = 200

    def test1(self):
        print("---test1---")

    def __test2(self):
        print("---test2---")

class B(A):
    pass

b = B()
b.test1()
#b.__test1() #私有方法并不会被继承
print(b.num1)
#print(b.__num2) #私有属性并不会被继承

第二种:

class A:
    def __init__(self):
        self.num1 = 100
        self.__num2 = 200

    def test1(self):
        print("---test1---")

    def __test2(self):
        print("---test2---")

    def test3(self):
        self.__test2()
        print(self.__num2)

class B(A):
    pass

b = B()
#b.test1()
#b.__test1() #私有方法并不会被继承
#print(b.num1)
#print(b.__num2) #私有属性并不会被继承
b.test3()

得到结果:

/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
---test2---
200

Process finished with exit code 0

第三种:

class A:
    def __init__(self):
        self.num1 = 100
        self.__num2 = 200

    def test1(self):
        print("---test1---")

    def __test2(self):
        print("---test2---")

    def test3(self):
        self.__test2()
        print(self.__num2)

class B(A):
    def test4(self):
        self.__test2()
        print(self.__num2)

b = B()
#b.test1()
#b.__test1() #私有方法并不会被继承
#print(b.num1)
#print(b.__num2) #私有属性并不会被继承
#b.test3()
b.test4()

如果是像第二种那样,在B的父类A里面调用test3()的函数,而test3()函数里面调用私有属性和私有方法,那么是可以得到结果的。
但是如果像第三种那样,在子类里面创建一个新的test4(),试图在这个函数里面调用父类里面的私有属性和私有方法,那么就是不被允许的。
专业一点来说,就是,如果调用的是父类中的公有方法,则可以在这个公有方法中访问父类中的私有属性和私有方法;但是如果在子类中实现了一个公有方法,那么这个方法是不能够调用继承的父类中的私有方法和私有属性。

  1. 多继承
class Base:
    def test(self):
        print("----Base")

class A(Base):
    def test1(self):
        print("----test1")

class B(Base):
    def test2(self):
        print("----test2")

#如果想要C()继承A()和B()两个父类的函数功能,那么只需要在括号里面把二者添加上就可以了
#当然,它本来就可以调用Base里面的函数功能
class C(A, B):
    pass

c = C()
c.test1()
c.test2()
c.test()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
----test1
----test2
----Base

Process finished with exit code 0

关于多继承的调用

class Base:
    def test(self):
        print("----Base")

class A(Base):
    def test(self):
        print("----A")

class B(Base):
    def test(self):
        print("----B")

class C(A, B):
    def test(self):
        print("----C")

c = C()
c.test()
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
----C

Process finished with exit code 0

值得注意的是,如果C里面有test(),那么它就会在调用完C之后就停止了;如果C里面没有test(),那么它就只会调用A的而不会A和B都调用,这里面可以用"print(C.__mro __)"来查看它实现的路径

  1. 多态
class Dog(object):
    def print_self(self):
        print("Hello world")

class Xiaotq(Dog):
    def print_self(self):
        print("Oh ha you")

def introduce(temp):
    temp.print_self()


dog1 = Dog()
dog2 = Xiaotq()

introduce(dog1)
introduce(dog2)
/Library/Frameworks/Python.framework/Versions/3.6/bin/python3.6 /Users/xinqi/PycharmProjects/test/test.py
Hello world
Oh ha you

Process finished with exit code 0

多态的含义就是,当你写完程序之后,我仅仅知道你是调用一个方法"temp.print_self()",但是我不确定它到底是调用基类的还是子类的。但是真正在程序执行的时候,我才根据当前的对象是谁,我才确定到底是调用Xiaotq()里面的方法还是调用Dog()里面的方法,这就是多态。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,980评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,422评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,130评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,553评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,408评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,326评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,720评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,373评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,678评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,722评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,486评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,335评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,738评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,283评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,692评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,893评论 2 335

推荐阅读更多精彩内容

  • Python 面向对象Python从设计之初就已经是一门面向对象的语言,正因为如此,在Python中创建一个类和对...
    顺毛阅读 4,203评论 4 16
  • 因为昨晚整夜心事重重,辗转难眠,今天早上难得起了个大早。 为了提神,我打算去洗浴室洗了把脸,推门而出的那一瞬间,寒...
    温时阅读 208评论 0 0
  • 大二的学生,挂科三门,四级,计算机。驾照,该有的都没有。也不知道是怎么了,浑浑噩噩就过了这么久,明天考试,现在凌晨...
    sunset妮妮阅读 407评论 13 6
  • 5月15日下午2:30,重庆理工大学第十届大学生心理素质拓展活动月开幕式在学生活动中心学术报告厅隆重举行。重庆市精...
    重庆理工大学阅读 532评论 0 0
  • 开学一周了,接受军训也一周了,和麻麻微信通话一周了,家里叫我姐姐的小家伙也想了我一周了。我也一样。 每早六点四十出...
    玩呐阅读 716评论 0 0