前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Python面向对象编程Day 26部分

Python面向对象编程Day 26部分

作者头像
py3study
发布2020-01-19 11:49:49
3990
发布2020-01-19 11:49:49
举报
文章被收录于专栏:python3python3python3

五大数据类型:整型、字符串、列表、元组、字典

多态

多态概念指出了对象如何通过他们共同的属性和动作来操作和访问,而不需考虑他们具体的类、多态表明了动态(又名:运行时)绑定的存在,允许重载及运行时类型确定和验证。多态是继承的改变和扩展的实现机制,即调用不同的类实例化得对象下的相同方法,实现的过程不一样。

(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())
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019-03-13 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 多态
  • 封装
  • 反射/自省
  • 动态导入模块
    •   双下划线开头(内置方法)的attr方法
    • 二次加工标准类型(包装)
      •  包装标准类型
        • 组合的方式完成授权
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档