2019-08-09_Note_Day15

面向对象

一、内置类属性

1. 声明类属性

声明类的时候系统自动添加的属性(可能是字段也可能是对象属性)
class Person:
    def __init__(self, name, gender, age):
        self.name = name
        self.gender = gender
        self.age = age

    def eat(self, food):
        print('{}在吃{}'.format(self.name, food))


p1 = Person('小明', '男', 18)

1)__name__

类的字段;类名.__name__:获取当前类名(字符串)
print(type(Person))  # <class 'type'>
print(Person.__name__)  # Person
print(type(Person.__name__))  # <class 'str'>

2)__doc__

类的字段;类名.__doc__:获取类的说明文档
print(Person.__doc__)
"""
    说明文档: 人类
    name - 名字
    gender - 性别
    age - 年龄
"""

3)__class

对象属性;对象.__class__:获取对象对应的类,返回的是类(和type功能相同)
print(p1.__class__)  # <class '__main__.Person'>

4)__dict__

对象属性;对象.__dict__:将对象中所有的属性和对应的值转换成一个字典中的键值对(一个对象一个字典)
类的字段;类.dict:将类转换成一个字典,字典中的元素是类中所有的字段和对应的值
print(p1.__dict__)  # {'name': '小明', 'gender': '男', 'age': 18}

print(Person.__dict__)
'''
{'__module__': '__main__', '__doc__': '\n\t说明文档: 人类\n\tname - 名字\n\tgender - 性别\n\tage - 年龄\n\t', '__init__': <function Person.__init__ at 0x1007ac400>, 'eat': <function Person.eat at 0x1007ac510>, '__dict__': <attribute '__dict__' of 'Person' objects>, '__weakref__': <attribute '__weakref__' of 'Person' objects>}
'''
定制当前对象的打印
a. 重写str方法,这个方法的返回值就是对应的打印结果(类型必须是字符串)
class Person:
...
    def __str__(self):
        return ''.join(['<', str(self.__dict__)[1:-1], '>'])


print(p1)  # <'name': '小明', 'gender': '男', 'age': 18>
b. 重写repr方法,这个方法的返回值就是对应的打印结果(类型必须是字符串)
class Person:
...
    def __repr__(self):
        return ''.join(['<', str(self.__dict__)[1:-1], '>'])


print(p1)  # <'name': '小明', 'gender': '男', 'age': 18>
注意:如果设置了slots的值,当前类的对象就不能使用dict属性

5)__module__

类的字段;类.__module__:获取当前类是在哪个模块中声明的(返回的是模块的名字)
print(Person.__module__)  # __main__
print(bool.__module__)  # builtins

6)__bases__

类的字段;类.__bases__:获取当前类的父类(返回的是一个元组)
print(Person.__bases__)  # (<class 'object'>,)

2.私有化

1)访问权限:公开、保护、私有

公开:公开的属性和方法在类的内部、外部可以使用,可以被继承
保护:保护的属性和方法在类的内部可以使用,外部不能使用, 可以被继承
私有:私有的属性和方法只能在在类的内部使用,外部不能使用, 也不能被继承


2)python中属性和方法的访问权限

python中类的所有属性和方法本质都是公开的;
私有化是假私有化,只是提示程序员这个属性或者方法在外部不要使用,也不要去继承
怎么私有化:在需要私有化的属性名或者方法名前加__(注意:不能以__结尾)
python私有化的原理:在私有的属性和方法前加了'_类名'

3.属性的getter和setter

1)什么是getter和setter

当我们需要在获取属性值之前对其进行其他操作,则需要对其添加getter;
当我们需要在给属性值之前对其进行其他操作,则需要对其添加setter

2)给属性添加getter

a. 属性命名时前面加下划线'_'
b. 在@property装饰器的后面声明一个对象方法

第一,将属性去掉下划线作为方法名
第二,方法除了self外不需要其他参数
第三,函数的返回值就是获取这个属性时得到的值

c. 在外部使用属性时,通过 '对象.不带下划线的属性名' 去使用
注意:获取属性指定值的时候, 就会自动去调用getter对应的方法

3)给属性添加getter

属性添加setter之前必须先添加getter
a. 保证属性名前有下划线
b. 在@getter名.setter装饰器的后面声明一个同名对象方法,需要一个self以外的参数,不需要返回值
c. 在外部使用属性时,通过 '对象.不带下划线的属性名' 去使用
注意:当给属性赋值时, 实质是调用setter对应的方法
class Person:
    def __init__(self, name, age, gender):
        self.name = name
        self._age = self.age = age
        self.__gender = self.gender = gender

    @property
    def age(self):
        print('年龄被访问!')
        return self._age

    @age.setter
    def age(self, value):
        print('年龄被修改!')
        if not isinstance(value, int):
            raise ValueError
        if not 0 <= value <= 120:
            raise ValueError
        self._age = value

    @property
    def gender(self):
        if self.__gender == 1:
            return 'male'
        elif self.__gender == 2:
            return 'female'
        else:
            return 'unknown'

    @gender.setter
    def gender(self, value):
        if value == 'male':
            self.__gender = 1
        elif value == 'female':
            self.__gender = 2
        else:
            self.__gender = 0

    def __repr__(self):
        return 'Person<name: {}, age: {}, gender: {}>'.format(self.name, self.age, self.gender)
练习:写一个矩形类
# 有属性长、宽、面积、周长
# 要求从生活的角度考虑


class WriteError(Exception):
    def __str__(self):
        return 'Modify read-only properties!'


class Rectangle:
    def __init__(self, length, width):
        self._length = self.length = length
        self._width = self.width = width
        self._area = self.length * self.width
        self._perimeter = 2 * (self.length + self.width)

    @property
    def length(self):
        return self._length

    @length.setter
    def length(self, value):
        if not (isinstance(value, int) or isinstance(value, float)) or value <= 0:
            raise ValueError
        self._length = value

    @property
    def width(self):
        return self._width

    @width.setter
    def width(self, value):
        if not (isinstance(value, int) or isinstance(value, float)) or value <= 0:
            raise ValueError
        self._width = value

    @property
    def area(self):
        self._area = self.length * self.width
        return self._area

    @area.setter
    def area(self, value):
        raise WriteError

    @property
    def perimeter(self):
        self._perimeter = 2 * (self.length + self.width)
        return self._perimeter

    @perimeter.setter
    def perimeter(self, value):
        raise WriteError

    def __repr__(self):
        return '<length:{}, width: {}, area: {}, perimeter: {}>'. \
            format(self._length, self._width, self.area, self.perimeter)


r1 = Rectangle(10, 20)
print(r1.area, r1.perimeter)  # 200 60
print(r1)  # <length:10, width: 20, area: 200, perimeter: 60>
r1.length = 20
print(r1.area)  # 400
print(r1)  # <length:20, width: 20, area: 400, perimeter: 60>
# r1.area = 100  # __main__.WriteError: Modify read-only properties!

二、类方法和静态方法

1. 类中的函数

类中的方法分为:对象方法、类方法和静态方法
对象方法 类方法 静态方法
如何声明 直接声明 声明在@classmethod后面 声明在@staticmethod后面
如何调用 对象.方法名调用 类名.方法名调用 类名.方法名调用
特点 有指向当前对象的self 有指向当前类的自带参数cls,这个参数在调用时不需要手动传参 没有默认参数,是声明在类中的一个普通方法
使用情景 如果实现的功能需要用到对象属性,就使用对象方法 如果实现的功能需要用到类(不需要对象属性),就使用类方法 如果实现的功能既不需要用到对象属性,也不需要类,就使用静态方法
class Person:
    num = 61

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

    def ear(self, food):
        # 对象能做的事情self都能做
        print('{}在吃{}'.format(self.name, food))

    @classmethod
    def func1(cls):
        # 类能做的事情cls都能做
        # a = cls('a', 18, 'b')
        # print(a)
        print('这是一个类方法!')

    @staticmethod
    def func2():
        print('这是一个静态方法!')

    def __repr__(self):
        return 'Person<name: {}, age: {}, gender: {}>' \
            .format(self.name, self.age, self.gender)


Person.func1()
Person.func2()

三、继承

1. 什么是继承

继承就是让子类直接拥有父类的属性和方法
子类:继承者
父类/超类:被继承者

2. 如何继承

1)语法

class 类名(父类1, 父类2, 父类3...):
    说明文档
    类的内容

2)说明:

():固定写法,如果省略相当于(object),声明类的时候如果没有写父类,默认继承object(object又称基类)
父类:一个类的父类可以有多个,但是一般情况下只有一个父类(支持多继承)
class Person:
    num = 61

    def __init__(self):
        self.name = '小明'
        self.age = 18
        self.gender = '男'
        self.__a = 10

    def eat(self, food='米饭'):
        print('{}在吃{}'.format(self.name, food))

    @classmethod
    def show(cls):
        print('人类的数量:{}'.format(cls.num))

    @staticmethod
    def func1():
        print('人类')

    def func2(self):
        print('我是{}'.format(self.name))


class Student(Person):
    num = 10

    def __init__(self):
        # 在子类中添加对象属性,需要先通过super().__init__()来继承父类的对象属性
        super().__init__()
        self.study_id = '001'
        self.class1 = 'py1904'

    # 重写
    @staticmethod
    def func1():
        print('学生')

    def func2(self):
        print('我是学生', end=', ')
        # 在子类中可以通过super().方法()可以调用父类的方法
        # 注意:super()只能在对象方法和类方法中使用
        super().func2()

    # 添加功能
    def study(self):
        print('{}在学习'.format(self.name))


stu1 = Student()
print(stu1.num)  # 61
print(stu1.name, stu1.age, stu1.gender)  # 小明 18 男
stu1.eat()  # 小明在吃米饭
Student.show()  # 人类的数量:61
print(stu1.__dict__)  # {'name': '小明', 'age': 18, 'gender': '男', '_Person__a': 10, 'study_id': '001', 'class1': 'py1904'}
stu1.func1()  # 学生
stu1.func2()  # 我是学生, 我是小明

3. 多继承

python中的类支持多继承(让一个类同时继承多个类)
多继承的时候子类只能继承第一个父类所有的属性和方法
后面的父类中只有字段和方法可以被继承
class Animal(object):
    num = 100

    def __init__(self):
        self.age = 0
        self.gender = 'female'

    @classmethod
    def func1(cls):
        print('动物类的类方法')

    def func2(self):
        print('动物类的对象方法')


class Fly(object):
    name = 'fly'

    def __init__(self):
        self.height = 100
        self.time = 5
        self.speed = 100

    def func2(self):
        print('飞行的对象方法')


class Bird(Animal, Fly):
    pass


bird = Bird()

# 字段都能继承
print(Bird.num, Bird.name)  # 100 fly

Bird.func1()  # 动物类的类方法
bird.func2()  # 飞行的对象方法

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

推荐阅读更多精彩内容