多态概念指出了对象如何通过他们共同的属性和动作来操作和访问,而不需考虑他们具体的类、多态表明了动态(又名:运行时)绑定的存在,允许重载及运行时类型确定和验证。多态是继承的改变和扩展的实现机制,即调用不同的类实例化得对象下的相同方法,实现的过程不一样。
(python中的标准类型就是多态概念的一个良好示范)
本质是要明确的区分内外。Python约定(并不是实际的限制)单下划线开头的变量为隐藏变量。python会给双下划线开头的变量重命名(前加上'_ClassName')
第一个层面的封装:类就是麻袋,这本身就是一种封装
第二个层面的封装:类中定义私有的,只在类的内部使用,外部无法访问(停留在约定层面)
第三个层面的封装(真正的封装):明确区分内外(内部可以直接使用,外部则不能),内部的实现逻辑,外部无法知晓,并且为封装到内部的逻辑提供一个访问接口(访问函数 )给外部使用
通过继承+多态在语言层面支持了归一化设计。不用面向对象的语言即class一样可以做归一化(比如泛文件概念),一样可以封装,只是使用面向对象语言可以直接用语言元素显示声明而已。
主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(也叫自省)。
四个可以实现自省的函数(适用于类和对象):
hasattr(object,name,default=None)判断object中有没有一个name字符串对应的方法或属性,没有打印False
getattr(object,name,default=None)没有则报错,如果default=None位置处写一个错误提示字符串,就不会报错了,而是输出这个字符串
setattr(object,key(字符串形式),value )添加或修改
delattr(object,key(字符串形式))相当于 del object.key
底层是基于反射去做的。
import dic
m=__import__('dic') #字符串形式导入模块,只能导入最顶级模块
print(m) #module 'm1'
module_t=__import__('m1.t') #实际引入的是m1
module_t.t #是在调用m1文件夹中的t文件
from m1.t import * #标红是因为没在环境变量中 导入不了函数名以下划线开头的函数
import importlib
m=importlib.import_module('m1.t') #这回直接定位到t
print(m) #module 'm1.t'
m.test1()调用t文件下的test1函数
双下划线开头(内置方法)的attr方法
属性不存在时触发__getattr__函数(最重要,用的比较多)
删除一个对象的属性时会触发__delattr__函数
__setattr__(self,key,value)设置时触发 函数内self.key=value这样会触发无限递归 应该这样设置self.__dict__[key]=value操作底层字典(del 什么.什么也会导致无限递归) 最后执行设属性操作可通过 f=函数名(参数) f.key=value这样设定
print(函数名.__dict__)和print(dir(函数名))都是用来打印属性方法的,但是后者更全面
1 class Foo:
2 def __init__(self,name):
3 self.name=name
4 def __getattr__(self, item):
5 print('你找的属性【%s】不存在' %item)
6 def __setattr__(self, k,v):
7 print('执行setattr',k,v)
8 if type(v) is str:
9 print('开始设置')
10 # self.k=v #触发__setattr__
11 self.__dict__[k]=v.upper()
12 else:
13 print('必须是字符串类型')
14 def __delattr__(self, item):
15 print('不允许删除属性【%s】' %item)
16 # print('执行delattr',item)
17 # del self.item
18 # self.__dict__.pop(item)
19
20 f1=Foo('alex')#触发__setattr__
21 # f1.age=18 #触发__setattr__
22 # print(f1.__dict__)
23 # print(f1.name)
24 # print(f1.age)
25 # print(f1.gender)
26 # print(f1.slary)
27 print(f1.__dict__)
28 del f1.name
29 print(f1.__dict__)
输出
执行setattr name alex
开始设置
{'name': 'ALEX'}
不允许删除属性【name】
{'name': 'ALEX'}
授权是包装的一个特性,包装一个类型通常是对已存在的类型的一些定制,修改或删除原有产品功能,其它的保持原样。授权的过程即所有更新的功能都是由新类的某部分处理,但已存在的功能就授权给对象的默认属性。
包装通过继承加派生实现。
实现授权的关键点就是覆盖__getattr__方法。
str()工厂函数 str一个类
包装标准类型
1 class List(list):
2 def append(self, p_object):
3 if type(p_object) is str:
4 # self.append(p_object)
5 super().append(p_object)
6 else:
7 print('只能添加字符串类型')
8
9 def show_midlle(self):
10 mid_index=int(len(self)/2)
11 return self[mid_index]
12
13
14 # l2=list('hell oworld')
15 # print(l2,type(l2))
16
17 l1=List('helloworld')
18 # print(l1,type(l1))
19 # print(l1.show_midlle())
20 l1.append(1111111111111111111111)
21 l1.append('SB')
22 print(l1)
输出
只能添加字符串类型
['h', 'e', 'l', 'l', 'o', 'w', 'o', 'r', 'l', 'd', 'SB']
组合的方式完成授权
1 import time
2 class FileHandle:
3 def __init__(self,filename,mode='r',encoding='utf-8'):
4 # self.filename=filename
5 self.file=open(filename,mode,encoding=encoding)
6 self.mode=mode
7 self.encoding=encoding
8 def write(self,line):
9 print('------------>',line)
10 t=time.strftime('%Y-%m-%d %X')
11 self.file.write('%s %s' %(t,line))
12
13 def __getattr__(self, item):
14 # print(item,type(item))
15 # self.file.read
16 return getattr(self.file,item)
17
18 f1=FileHandle('a.txt','w+')
19 # print(f1.file)
20 # print(f1.__dict__)
21 # print('==>',f1.read) #触发__getattr__
22 # print(f1.write)
23 f1.write('1111111111111111\n')
24 f1.write('cpu负载过高\n')
25 f1.write('内存剩余不足\n')
26 f1.write('硬盘剩余不足\n')
27 # f1.seek(0)
28 # print('--->',f1.read())
0