一、三大编程范式
二、编程进化论
三、面向对象设计与面向对象编程
# 面向对象设计
def dog(name,gender,type): # 相当于类的概念
# 动作
def jiao(dog):
print("一条 %s 正在叫" %dog["name"])
def chi(dog):
print("一条 %s 正在吃骨头"%dog["name"])
# 属性 特征
def init(name,gender,type): # 初始化函数
dog1 = {
"name": name,
"gender": gender,
"type": type,
"jiao": jiao ,
"chi": chi
}
return dog1
return init(name,gender,type)
d1 = dog("xiaoming","mu","zangao") # 相当于一个对象
d2 = dog("xiaogang","mu","zangao")
print(d1)
d1["chi"](d1)
d2["jiao"](d2)
四、小结
五、类和对象
class Teacher:
‘这是一个教师类’
def __init__(self,name,age,job_in,teach_subjects): # 实例化
self.name = name # self 实例本身
self.age = age # 相当于teacher1.name = name
self.job_in = job_in
self.teach_subjects = teach_subjects
def teach(self):
print("%s 在 %s 教 %s" %(self.name,self.job_in,self.teach_subjects))
def talk(self,name,content):
print("%s 在和 %s 讨论 %s"%(self.name,name,content)) teacher1 = Teacher("lhf",33,"oldboy","python") # 对象# 就相当于,理论上teacher1 = Teacher.__init__(teacher1,name,age,job_in,teach_subjects) teacher2 = Teacher("wusir",28,"oldboy","python") # 对象 teacher1.teach() # 调用教学方法
teacher2.talk("小明","如何学好python")
5.1 属性
类是用来描述一类事物,类的对象指的是这一类事物中的一个个体
是事物就要有属性,属性分为
1:数据属性:就是变量
2:函数属性:就是函数,在面向对象里通常称为方法
注意:类和对象均用点来访问自己的属性
类的数据属性
class ChinesePeople:
government='共产_党' print(ChinesePeople.government)
类的函数属性
class Chinese:
'这是一个中国人的类'
goverment = "共产_党" # 数据属性
def __init__(self,name,age,gender): # 创建一个实例对象
print(" 我开始了")
self.mingzi = name
self.nianling = age
self.xingbei = gender
def gongfu(): # 函数属性 又称方法
print("会功夫")
def xihuanchi(self):
print("%s 讲究吃"%self.mingzi)
p1 = Chinese("小明",18,"female") # 对象
print(p1.__dict__) # 相当于init的返回值 执行__init__函数 {'mingzi': '小明', 'xingbei': 'female', 'nianling': 18}
print(p1.mingzi) # 小明
print(p1.goverment) #共产_党
print(Chinese.__dict__)# {'__init__': <function Chinese.__init__ at 0x0000016DDEDCF378>, '__dict__': <attribute '__dict__' of 'Chinese' objects>, 'gongfu': <function Chinese.gongfu at 0x0000016DDEDCF400>, 'goverment': '共产_党', 'xihuanchi': <function Chinese.xihuanchi at 0x0000016DDEDCF488>, '__module__': '__main__', '__doc__': '这是一个中国人的类', '__weakref__': <attribute '__weakref__' of 'Chinese' objects>}
Chinese.gongfu() #会功夫
Chinese.xihuanchi(p1) # 小明 讲究吃
p1.xihuanchi() #小明 讲究吃
查看类属性
我们定义的类的属性到底存到哪里了?有两种方式查看
dir(类名):查出的是一个名字列表
类名.__dict__:查出的是一个字典,key为属性名,value为属性值
特殊的类属性
# class dog :
# '这是定义一个狗的对象'
# name = "藏獒"
# def jiao(self):
# print("一条 %s 正在汪汪汪" %self)
# dog.jiao("藏獒")
# print(dir(dog)) # 显示结果是一个列表,包含类(包含内建属性在内的)所有的属性名
# print(dog.__dict__) # 显示结果是一个字典,包含类的所有属性:属性值
#
# print(dog.__name__)# 类C的名字(字符串)
# print(dog.__doc__)# 类C的文档字符串
# print(dog.__base__)# 类C的第一个父类(在讲继承时会讲)
# print(dog.__bases__)# 类C的所有父类构成的元组(在讲继承时会讲)
# print(dog.__dict__)# 类C的属性
# print(dog.__module__)# 类C定义所在的模块
# print(dog.__class__)# 实例C对应的类(仅新式类中)
5.2 类属性与对象(实例)属性
# 类属性的增删改查 # class Chinese:
# country='China'
# def __init__(self,name):
# self.name=name
#
# def play_ball(self,ball):
# print('%s 正在打 %s' %self.name,ball)
#
# def say_word(self,word):
#
# print('%s 说 %s' %(self.name,word))
# 数据属性的增删改查
# 查看类的属性
# print(Chinese.country)
# 修改
# Chinese.country = "English"
# print(Chinese.country)
# 增加
# Chinese.location = "Asia"
# print(Chinese.__dict__)
# print(Chinese.location)
# # 删除
# del Chinese.location
# print(Chinese.__dict__) # 实例属性的增删改查
# 查看
# p1 = Chinese("xiaoming")
# print(p1.__dict__)
# # 查看
# print(p1.name)
# # 删除
# del p1.name
# print(p1.__dict__)
# # 增加
# p1.gender = "man"
# print(p1.__dict__)
# # 修改
# p1.gender = "woman"
# print(p1.__dict__) # 函数属性的增加修改
# class Chinese:
# country='China'
# def __init__(self,name):
# self.name=name
#
# def play_ball(self,ball):
# print('%s 正在打 %s' %(self.name,ball))
#
# def say_word(self,word):
#
# print('%s 说 %s' %(self.name,word))
# # 增加
# def eat_food(self,food):
# print("%s 正在吃 %s" %(self.name,food))
# Chinese.eat = eat_food
# print(Chinese.__dict__)
# p1 = Chinese("xiaoming")
# # print(p1.__dict__)
# p1.eat("fan")
# 注意事项
#
# class Chinese:
# country = "China"
# def __init__(self,name):
# self.name = name
# def play_ball(self,ball):
# print("%s 正在打 %s" %(self.name,ball))
# p1 = Chinese('alex')
# print(p1.country)
# p1.country = "日本" # 相当于在实例里面创建了个新的
# print(p1.__dict__) # {'name': 'alex', 'country': '日本'}
# print("类的——",Chinese.country)
# print("实例的",p1.country) # 2、不调用 . 直接找全局变量
# country = "中国"
# class Chinese:
# country = "China"
# def __init__(self,name):
# self.name = name
# print(country) # 中国
# def play_ball(self,ball):
# print("%s 正在打 %s" %(self.name,ball))
# p1 = Chinese('alex')
# print(p1.country) # China
六 静态属性、类方法、静态方法
方法包括:普通方法、静态方法和类方法,三种方法在内存中都归属于类,区别在于调用方式不同。
普通方法:由对象调用;至少一个self参数;执行普通方法时,自动将调用该方法的对象赋值给self;
类方法:由类调用; 至少一个cls参数;执行类方法时,自动将调用该方法的类复制给cls;classmethod
静态方法:由类调用;无默认参数;staticmethod
class Room:
x = 1
def __init__(self,owner,name,length,width,height):
self.owner = owner
self.name = name
self.length = length
self.width = width
self.height = height
@property # 把函数属性封装为(静态属性)(数据属性)
def cal_volume(self):
return self.length * self.height * self.width @classmethod # 类方法 : 专门给类使用,(数据属性)与实例无关,类方法只能访问类相关的属性,不能访问实例属性
def tell_info(cls,name):
print(cls)
print(cls.x,name)
@staticmethod # 静态方法只是名义上的归属类的管理,不能使用类变量和实例变量,是类的工具包
def text(a,b,c):
print(a,b,c)
Room.text(1,2,3)
Room.tell_info("xiaoming")
r1 = Room("小明","豪华别墅",10,20,30)
print(r1.cal_volume)
# 静态属性
class Room:
tag = 1 def __init__(self, owner, name, width, length, high):
self.owner = owner
self.name = name
self.width = width
self.length = length
self.high = high @property # 在调用函数的时候把括号去掉, 相当于调用属性, 像是(把函数属性,变成了数据属性)这样讲是不准确的
def cla_area(self):
# print("%s 住的 %s 的总面积 %s" % (self.owner, self.name, self.width * self.length))
return "%s 住的 %s 的总面积 %s" % (self.owner, self.name, self.width * self.length) def test(self):
print('from test', self.name) @classmethod # 只和类有关和实例没有关系 ,能调用数据属性, 类级别的调用 #定义备选构造方法
def test1(cls, x):
print('from test1', cls.tag, x) @staticmethod # 类的工机具包, 不和实例绑定,不和类绑定, ,静
# 态方法就是普通的函数,只是碰巧在类的定义体中,而不是在模块层定义
def test2():
print("aaaaa") r1 = Room("主卧", 'sk', 20, 10, 100)
r2 = Room("主卧", 'ck', 10, 10, 100)
# print(r1.cla_area)
# print(r2.cla_area) # 类方法 print(Room.tag)
Room.test1("xiao") Room.test2()
七 组合
# 定义一个人的类,人有头,躯干,手,脚等数据属性,这几个属性又可以是通过一个类实例化的对象,这就是组合
# 用途:
# 1:做关联
# 2:小的组成大的
class Hand:
pass
class Foot:
pass
class Trunk:
pass
class Head:
pass class Person:
def __init__(self,id_num,name,hand,foot,trunk,head):
self.id_num=id_num
self.name=name
self.hand=Hand()
self.foot=Foot()
self.trunk=Trunk()
self.head=Head()
class School:
'定义一个学校的类'
def __init__(self,name,addr):
self.name = name
self.addr = addr
def recruit_student(self):
print("%s 正在招生" %self.name)
class Course:
'定义一个课程的类'
def __init__(self,name,price,period,school):
self.name = name
self.price = price
self.period = period
self.school = school
s1 = School("oldboy","北京") # 为学校类创建一个对象
s2 = School("oldboy","南京")
s3 = School("oldboy","东京")
msg = '''
1、老男孩 北京校区
2、老男孩 南京校区
3、老男孩 东京校区
'''
while True:
print(msg)
menu = {
"":s1,
"":s2,
"":s3
} # 对应学校的类
choice = input("选择学校》》》")
school_obj = menu[choice] # 就相当于是 school_obj = s1
name = input("课程名》》》")
price = input("课程费用》》》")
period = input("课程周期》》》")
new_course = Course(name,price,period,school_obj) # 最后传的是s1
print("课程 【%s】 属于【%s】 学校 %s 校区"%(new_course.name,new_course.school.name,new_course.school.addr))
选学校和课程
八 面向对象编程三大特性
8.1 继承
一 什么是类的继承?
类的继承跟现实生活中的父、子、孙子、重孙子、继承关系一样,父类又称为基类。
python中类的继承分为:单继承和多继承
class Base:
pass
class Father:
pass
class Son(Father): #单继承
pass
class Son(Base,Father): #多继承
pass
二 子继承到底继承了父类的什么属性?
class Dad:
'这是父类'
money = 50000
def __init__(self,name):
self.name = name
def hit_son(self):
print("%s 正在打孩子"%self.name)
class Son(Dad):
'这是一个儿子的类'
money = 50000000 # 子类继承了父类的所有的属性,子类中定义了就不会再找父类中的方法
print(Dad.__dict__)
print(Son.__dict__) s1 = Son("小明")
s1.hit_son()
print(s1.money)
print(Dad.money)
三 什么时候用继承?
1.当类之间有显著不同,并且较小的类是较大的类所需要的组件时,用组合比较好
例如:描述一个机器人类,机器人这个大类是由很多互不相关的小类组成,如机械胳膊类、腿类、身体类、电池类
2.当类之间有很多相同的功能,提取这些共同的功能做成基类,用继承比较好
例如
猫可以:喵喵叫、吃、喝、拉、撒
狗可以:汪汪叫、吃、喝、拉、撒
四 继承同时具有两种含义
import abc
class All_file(metaclass=abc.ABCMeta):
@abc.abstractmethod
def read(self):
pass
@abc.abstractmethod
def write(self):
pass
class Disk(All_file):
def read(self):
print("disk read")
def write(self):
print("disk write")
class Cdrom(All_file):
def read(self):
print("cdrom read")
def write(self):
print("cdrom write")
class Mem(All_file):
def read(self):
print("mem read")
def write(self):
print("mem write")
m1 = Mem()
m1.read()
实践中,继承的第一种含义意义并不很大,甚至常常是有害的。因为它使得子类与基类出现强耦合。
继承的第二种含义非常重要。它又叫“接口继承”。
接口继承实质上是要求“做出一个良好的抽象,这个抽象规定了一个兼容接口,使得外部调用者无需关心具体细节,可一视同仁的处理实现了特定接口的所有对象”——这在程序设计上,叫做归一化。
归一化使得高层的外部使用者可以不加区分的处理所有接口兼容的对象集合——就好象linux的泛文件概念一样,所有东西都可以当文件处理,不必关心它是内存、磁盘、网络还是屏幕(当然,对底层设计者,当然也可以区分出“字符设备”和“块设备”,然后做出针对性的设计:细致到什么程度,视需求而定)。
五 继承顺序
class A:
def test(self):
print("A")
class B(A):
def test(self):
print("A") class C(A):
def test(self):
print("A") class D(B):
def test(self):
print("A") class E(C):
def test(self):
print("A") class F(D,E):
def test(self):
print("A")
f1 = F()
f1.test()
print(F.__mro__) # 继承顺序就是mro列表定义的顺序
终极解密:python到底是如何实现继承的,对于你定义的每一个类,python会计算出一个方法解析顺序(MRO)列表,这个MRO列表就是一个简单的所有基类的线性顺序列表
为了实现继承,python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。
而这个MRO列表的构造是通过一个C3线性化算法来实现的。我们不去深究这个算法的数学原理,它实际上就是合并所有父类的MRO列表并遵循如下三条准则:
1.子类会先于父类被检查
2.多个父类会根据它们在列表中的顺序被检查
3.如果对下一个类存在两个合法的选择,选择第一个父类
六 子类中调用父类方法
子类继承了父类的方法,然后想进行修改,注意了是基于原有的基础上修改,那么就需要在子类中调用父类的方法
class Vehicle:
'定义交通工具类'
def __init__(self,name,speed,load,power):
country = "China"
self.name = name
self.speed = speed
self.load = load
self.power = power
def run(self):
print("开动啦。。。") class Subway(Vehicle):
def __init__(self,name,speed,load,power,line):
# Vehicle.__init__(self,name,speed,load,power) # 调用父类的实例
# super().__init__(name,speed,load,power)
super(Subway,self).__init__(name,speed,load,power)
self.line = line
def run(self):
# Vehicle.run(self) # 调用父类的函数属性
super(Subway,self).run()
print("地铁%s号线欢迎您"%self.line)
line13 = Subway("中国地铁","180m/s","1000/车厢","电",13)
line13.run()
8.2 多态
什么是多态:
类的继承有两层意义:1.改变 2.扩展
多态就是类的这两层意义的一个具体的实现机制 即,调用不同的类实例化得对象下的相同的方法,实现的过程不一样
python中的标准类型就是多态概念的一个很好的示范如(str.__len__(),list.__len__(),tuple.__len__可以len(str),len(list),len(tuple)
class H20:
def __init__(self,name,tempterature):
self.name = name
self.tempterature = tempterature
def convert(self):
if self.tempterature <0:
print("温度太低变成冰")
elif self.tempterature >0 and self.tempterature<100:
print("温度在0—100之间是水")
elif self.tempterature>100:
print("温度太高变成蒸汽了蒸汽")
class Ice(H20):
pass
class Steam(H20):
pass
class Water(H20):
pass i = Ice("冰",-10)
s = Steam("水蒸气",200)
w = Water("水",80) def func(obj):
obj.convert()
func(i)
func(s)
func(w)
8.3 封装
第一个层面的封装:类就是麻袋,这本身就是一种封装
例:略
第二个层面的封装:类中定义私有的,只在类的内部使用,外部无法访问
python不依赖语言特性去实现第二层面的封装,而是通过遵循一定的数据属性和函数属性的命名约定来达到封的效果
约定一:任何以单下划线开头的名字都应该是内部的,私有的
################### 单 _ 情况
class People:
_star='earth'
def __init__(self,id,name,age,salary):
self.id=id
self.name=name
self._age=age
self._salary=salary def _get_id(self):
print('我是私有方法啊,我找到的id是[%s]' %self.id)
print(People._star)
p1=People('','alex',28,10) print(p1._age,p1._salary)
p1._get_id() 问题?为什么单杠情况还能被调用
单 _ 情况
python并不会真的阻止你访问私有的属性,模块也遵循这种约定,如果模块名以单下划线开头,那么from module import *时不能被导入,但是你from module import _private_module依然是可以导入的
其实很多时候你去调用一个模块的功能时会遇到单下划线开头的(socket._socket,sys._home,sys._clear_type_cache),这些都是私有的,原则上是供内部调用的
约定二:双下划线开头的名字
# ############ 双 __情况 ##############
class People:
__star='earth' # 双下划线开头
def __init__(self,id,name,age,salary):
self.id=id
self.name=name
self.__age=age
self._salary=salary
def _get_id(self):
print('我是私有方法啊,我找到的id是[%s]' %self.id)
print(People.__dict__)#__star存到类的属性字典中被重命名为_People__star
p1=People('','alex',18,10) print(p1.__dict__)#__age存到类的属性字典中被重命名为_People__age # print(People.star) # 这里不能被访问
# print(p1.age) # 这里不能被访问 print(p1._People__age) #可以通过这种方式访问 __age存到类的属性字典中被重命名为_People__age
双 __情况
双下滑线开头的属性在继承给子类时,子类是无法覆盖的(原理也是基于python自动做了双下滑线开头的名字的重命名工作)
第三层面的封装:明确区分内外,内部的实现逻辑,外部无法知晓,并且为封装到内部的逻辑提供一个访问接口给外部使用
九 面向对象的优点
从编程进化论我们得知,面向对象是一种更高等级的结构化编程方式,它的好处就两点
1:通过封装明确了内外,你作为类的缔造者,你是上帝,上帝造物的逻辑你无需知道(你知道了你tm也成上帝了),上帝想让你知道的你才能知道,这样就明确了划分了等级,物就是调用者,上帝就是物的创造者
2:通过继承+多态在语言层面支持了归一化设计
注意:不用面向对象语言(即不用class),一样可以做归一化(如老掉牙的泛文件概念、游戏行业的一切皆精灵),一样可以封装(通过定义模块和接口),只是用面向对象语言可以直接用语言元素显式声明这些而已;而用了面向对象语言,满篇都是class,并不等于就有了归一化的设计。甚至,因为被这些花哨的东西迷惑,反而更加不知道什么才是设计
十 python中关于OOP的常用术语
抽象/实现
抽象指对现实世界问题和实体的本质表现,行为和特征建模,建立一个相关的子集,可以用于 绘程序结构,从而实现这种模型。抽象不仅包括这种模型的数据属性,还定义了这些数据的接口。
对某种抽象的实现就是对此数据及与之相关接口的现实化(realization)。现实化这个过程对于客户 程序应当是透明而且无关的。
封装/接口
封装描述了对数据/信息进行隐藏的观念,它对数据属性提供接口和访问函数。通过任何客户端直接对数据的访问,无视接口,与封装性都是背道而驰的,除非程序员允许这些操作。作为实现的 一部分,客户端根本就不需要知道在封装之后,数据属性是如何组织的。在Python中,所有的类属性都是公开的,但名字可能被“混淆”了,以阻止未经授权的访问,但仅此而已,再没有其他预防措施了。这就需要在设计时,对数据提供相应的接口,以免客户程序通过不规范的操作来存取封装的数据属性。
注意:封装绝不是等于“把不想让别人看到、以后可能修改的东西用private隐藏起来”
真正的封装是,经过深入的思考,做出良好的抽象,给出“完整且最小”的接口,并使得内部细节可以对外透明
(注意:对外透明的意思是,外部调用者可以顺利的得到自己想要的任何功能,完全意识不到内部细节的存在)
合成
合成扩充了对类的 述,使得多个不同的类合成为一个大的类,来解决现实问题。合成 述了 一个异常复杂的系统,比如一个类由其它类组成,更小的组件也可能是其它的类,数据属性及行为, 所有这些合在一起,彼此是“有一个”的关系。
派生/继承/继承结构
派生描述了子类衍生出新的特性,新类保留已存类类型中所有需要的数据和行为,但允许修改或者其它的自定义操作,都不会修改原类的定义。
继承描述了子类属性从祖先类继承这样一种方式
继承结构表示多“代”派生,可以述成一个“族谱”,连续的子类,与祖先类都有关系。
泛化/特化
基于继承
泛化表示所有子类与其父类及祖先类有一样的特点。
特化描述所有子类的自定义,也就是,什么属性让它与其祖先类不同。
多态
多态的概念指出了对象如何通过他们共同的属性和动作来操作及访问,而不需考虑他们具体的类。
多态表明了动态(又名,运行时)绑定的存在,允计重载及运行时类型确定和验证。
举例:
水是一个类
不同温度,水被实例化成了不同的状态:冰,水蒸气,雾(然而很多人就理解到这一步就任务此乃多态,错,fuck!,多态是运行时绑定的存在)
(多态体现在由同一个类实例化出的多个对象,这些对象执行相同的方法时,执行的过程和结果是不一样的)
冰,水蒸气,雾,有一个共同的方法就是变成云,但是冰.变云(),与水蒸气.变云()是截然不同的两个过程,虽然调用的方法都一样
自省/反射
自省也称作反射,这个性质展示了某对象是如何在运行期取得自身信息的。如果传一个对象给你,你可以查出它有什么能力,这是一项强大的特性。如果Python不支持某种形式的自省功能,dir和type内建函数,将很难正常工作。还有那些特殊属性,像__dict__,__name__及__doc__