类和对象

面向对象
封装
继承
多态

程序 = 数据 + 功能
模块可以存放 数据 和 功能
函数是存放 功能 的,不调用不执行。
列表,存储 数据 和 方法(函数名),

把 数据 和 功能 整合起来,分为一类的编程思想就叫面向对象编程

类和对象
类:class

先定义类,再调用类产生对象

定义:定义阶段就运行,产生类的名称空间
__dict__:查看名称空间

class Student:  驼峰体
    ...  数据属性和方法属性

    调用类实例化对象的时候,运行,对对象进行初始化
    必须返回None,不能返回别的值
    def __init__(self,x,y,z):  
        self.x = x

egon = Student(x,y,z)  实例化对象
egon.x = 1
Student.x

产生空对象,调用__init__方法初始化对象,返回初始化的对象
属性或对象先从对象中找,然后再去类中找,去父类,,,
从自身开始查找,逐级往父级查找对应属性或方法,直到根对象Object.prototype,没有返回undefined

类中的公共属性:

直接改类中的属性的值,所有对象的相应的值都改了,
如果改的是对象中的值,那么相当于重新创建了自己的值,不影响类中的值,和其它实例化的值

def __init__(self,x,y,z):
    Student.count += 1

bound method 绑定方法:绑定给谁就操作谁

绑定方法地址里存的是啥?

school_shanghai = School('上海校区','虹桥火车站')
print(School.add_class)
print(school_shanghai.add_class)

<function School.add_class at 0x0000000000D73400>
<bound method School.add_class of <__main__.School object at 0x0000000000D70978>>

底层就是绑定给对象的名称空间,最终还是指向原来类的名称空间中的方法,
相当于给这个对象做了一个标记,它在调用的时候就给它临时绑定一下
是指向类中的函数的,地址不同,因为绑定方法做了进一步的处理,与普通函数不同


内置方法

方法

class Student:
    def choice():
        pass
Student.choice()  可以
stu.choice()  不可以

    def xxx(self):
        pass
Student.xxx(stu)  类调用自己的方法时,要注意对象参数


__dict__ : 名称空间
是个字典,注意字典的属性,没有某个值就进行添加

两个类的关联,一次即可

封装:
把各种有关联的属性,整合为一体,称为封装

将封装的属性隐藏

私有属性:不能外部直接调用,可以内部直接调用
__x
def __func(self):
    pass

类定义阶段:名称空间里的变量加前缀_Foo__x,_Foo__f1,统一变形,所以内部直接调用,如是访问可以
只是名字的变形,只在类定义,检测类体代码的语法的时候发生变形,之后的,不会发生变形

注意:!!!在哪个类里定义的私有属性,那么前缀就是哪个类名,注意类名前面是一个 _ ,
!!!
在遇到类的继承的时候,私有变量的引用,要注意父类的私有属性在子类里,要不通过父类提供的接口使用,要不就 父类._父类名__方法名!!!
不能在子类里直接self.__方法 调用,调用不到,因为名字在父类里已经变形

加强对属性的控制,不能够在类外随便操作,提供接口给外部进行访问;隔离复杂度

__init__ 内置属性:python自带的属性
__x 私有属性:自定义的私有属性

property:是一个装饰器,把函数的调用方式,改成变量的调用方式,得到的值是方法的return值

name = property(get_name,set_name,del_name)

将三个方法定义为最终的名字,然后不同的功能加上不同的装饰器,实现不同的调用方式,类似于一个变量的调用方式

@property
@name.setter
@name.deleter

继承
定义继承类的时候,什么是什么的类,就是有一定语言意义上的继承
创建新类的方式:子类,派生类;父类,基类,超类;
python支持多继承
class Sub(Parents):
    pass

class Sub(Parents,Parents1):
    pass

python2:
新式类:继承object类的子类等子类
经典类:没有继承object类的子类等子类
__bases__:看他的继承object情况
python3:都是新式类

子类没有的东西去父类找
def __init__(self,a,b):
    父类.__init__(self,a)
    self.b = b

菱形继承:
属性查找优先级:D.mro(),看类的顺序,mro列表
object类一定是最后找

非菱形继承:

python3:深度优先

python2 的新式类中有mro()

两个类,都是深度优先

菱形继承:

python3:

新式类:‘广度优先’

python2:

经典类:深度优先(不会重复找最终的类,这个类不是object类),第一次就找最终的类
新式类:‘广度优先’,最后找最终的类

注意:json和pickle不能追加序列化内容!!!

派生:子类产生新东西
子类独有,子类覆盖使用父类的功能,子类在用父类功能的基础上进行拓展

mixins机制:
多继承的正确使用方式:
提升多继承的可读性
mixin加入到混入类的名字后缀,表示将这个类混入加入一些功能
或者able或ible结尾

归属含义的父类只能有一个

Teacher.__init__(self,name,age,sex)

super(Teacher,self).__init__(name,age,sex)
super产生特殊的对象,参照当前类的mro,去第二个mro往后找,
在Python2中super的使用需要完整地写成super(自己的类名,self) ,而在python3中可以简写为super()。
# super(Teacher,self).__init__(name,age,sex)
# super().__init__(name,age,sex) # 调用的是方法,自动传入对象

组合

在一个类中以另外一个类的对象作为数据属性,称为类的组合。

多态

同一种事物有多种形态,就叫多态
多态带来的特性:多态性
不考虑对象具体类型的情况下,使用对象
都是动物类,所以都会叫
不同的类型可以有相同的方法



鸭子类型:推崇鸭子类型,而不是继承
linux 一切皆文件
把调用方式都变成文件的形式,也就是提供读和写的接口来使用

import abc

class Animal(metaclass = abc.ABCMeta):
    @abc.abstractmethod
    def say(self):
        pass

抽象基类:必须有父类Animal这个类的属性,统一所有子类的标准,多可以少不行
不能实例化Animal


绑定方法:
绑定给类:自动传入类参数
绑定给对象:自动传入对象参数

class Mysal:

    # 绑定给对象
    def __init__(self,ip,port):
        pass

    # 绑定给类
    @classmethod  # self 默认传入类
    def func(cls):  # 返回一个实例化的对象,从文件读取的参数,可以多次使用,           return cls(settings.IP,settings.PORt)  # 但是调用起来简单

绑定给类的方法:提供 生成对象的方法的接口,让实例化变简单,因为实例化有些参数是长期固定的,比如从配置文件settings中读取的

所以

mysql = Mysql.func()

    # 非绑定方法,静态方法,不绑定任何参数self,也可以在类里,function
    @staticmethod
    def a():
        pass

posted @ 2020-04-07 18:23  pythoner_wl  阅读(127)  评论(0编辑  收藏  举报