python全栈开发day23-面向对象高级:反射(getattr、hasattr、setattr、delattr)、__call__、__len__、__str__、__repr__、__hash__、__eq__、isinstance、issubclass
一、今日内容总结
1.反射
使用字符串数据类型的变量名来操作一个变量的值。
#使用反射获取某个命名空间中的值,
#需要
#有一个变量指向这个命名空间
#字符串数据类型的名字
#再使用getattr获取值,
#如果变量能直接获取结果
#如果是函数,只能拿到内存地址,加上括号就是执行
#使用类名反射: 静态属性、类方法、静态方法
#使用对象名反射: 对象属性、绑定对象方法
#使用模块名反射: 变量、函数、类
#在自己所在文件中反射全局变量: getattr(sys.modules[__name__],'要反射的名字')
#getattr
#hasattr
#setattr
#delattr
# import mmm
# print(mmm.a) # 直接使用模块调用模块中的变量
# print(getattr(mmm,'a')) # 使用反射调用模块中的变量a
# print(getattr(mmm,'b')) # 使用反射调用模块中的变量b
# mmm.wahaha() # 直接使用模块调用模块中的函数
# getattr(mmm,'wahaha')() # 使用反射调用模块中的函数 # print(mmm.QQxing) # 直接使用模块获取类的内存地址
# taibai = mmm.QQxing('光头') # 实例化模块中的类
# taibai.ADCa() # 实例调用方法
# cls = getattr(mmm,'QQxing') # 使用反射获取模块中的类
# obj = cls('光头') # 使用反射得到的类进行实例化
# print(obj)
# getattr(obj,'ADCa')() # 使用反射调用类中的方法 # hasattr ******
# getattr ******
# setattr ***
# delattr * # import mmm class A:
Name1 = 'alex'
Name2 = 'egon'
Name3 = 'guangtou'
def __init__(self,name):
self.name = name
def func(self):
print('in func',self.name) # A.Country = '中国' # 增
# A.Name1 = 'alex_sb' # 改
# # setattr
# print(A.__dict__)
# setattr(A,'Country','印度')
# print(A.__dict__)
# # 'Role' = 'Person'
# setattr(A,'Role','Person')
# print(A.__dict__)
# delattr(A,'Name1')
# print(A.__dict__)
2.__call__
# 对象加上(),可以触发这个类的__call__方法。
class Call_test:
def __call__(self):
print('I AM CALL') c1 = Call_test()
c1()
3.__len__
# 内置函数len函数是依赖于类中的__len__()
#__len__(self):
class Wahaha:
def __init__(self, num, age, sex):
self.num = num
self.age = age
self.sex = sex def __len__(self):return len(self.__dict__) obj = Wahaha(3, 12, 'male')
print(len(obj))
4.__str__和__repr__
# 在没有实现__str__时,__repr__可以完全替代__str__
# 在没有实现__repr__时,__str__不能替代__repr__
# 在两者都实现的情况下:
#__str__方法和str()、%s、print() 等息息相关
#__repr__方法和repr()、%r等息息相关
# format_dict={
# 'nat':'{obj.name}-{obj.addr}-{obj.type}',#学校名-学校地址-学校类型
# 'tna':'{obj.type}:{obj.name}:{obj.addr}',#学校类型:学校名:学校地址
# 'tan':'{obj.type}/{obj.addr}/{obj.name}',#学校类型/学校地址/学校名
# }
# class School:
# def __init__(self,name,addr,type):
# self.name=name
# self.addr=addr
# self.type=type
#
# def __format__(self, format_spec): # format_spec = 'nat'
# if not format_spec or format_spec not in format_dict:
# format_spec='nat'
# fmt=format_dict[format_spec] # fmt = '{obj.name}-{obj.addr}-{obj.type}'
# return fmt.format(obj=self)
#
# s1=School('oldboy1','北京','私立')
# print(format(s1,'nat'))
# print(format(s1,'tna'))
# print(format(s1,'tan'))
# print(format(s1,'asfdasdffd'))
# class A:pass
# o=A()
# o.name = 'a'
# o.addr = 'b'
# o.type = 'c'
# print('{obj.name}-{obj.addr}-{obj.type}'.format(obj = o))
5.__hash__ #hash()在优化内存中起到作用
6.__eq__
# 是判断值是否相等的时候依赖__eq__的结果
# 如果没有定义__eq__两个对象的值是否相等,会默认根据他们的内存地址是否相等判断
# Person class
# name age sex
# 两个对象 的name 和 sex相同,那么这两个对象就是相同的对象
# 对100个对象进行去重 class Person:
def __init__(self, name, age, sex):
self.name = name
self.age = age
self.sex = sex def __hash__(self):
return hash(self.name+self.sex) def __eq__(self, other):
if self.name==other.name and self.sex == other.sex:
return True
else:return False def __repr__(self):
return '|'.join([self.name, str(self.age), self.sex]) obj_li = []
for i in range(100):
obj_li.append(Person('alex', i, 'male')) print(obj_li)
print(set(obj_li)) # hash是否相等,值是否相等 __hash__值相等,__eq__值也相等
一个面试题深入了解__hash__和__eq__
# set()集合去重的内部机制:先判断hash值是否相同,如不同,直接重新开辟空间存储,如相等,再通过__eq__判断值是否相同,相同则直接丢弃,不相同,则在附近开辟内存空间存储。
7.isinstance和issubclass
# isinstance 是判断一个对象是否属于某个类或其父类
#issubclass是判断一个类是否是另一个类的子类,注意前后顺序,子类在前,父类在后。
二、预习和扩展
1、__del__
# 析构方法,当对象在内存中释放时,会自动触发执行。
#一个对象有用户级与内核级两种资源
当程序结束时,python只会回收自己的内存空间,即用户态内存,而操作系统的资源则没有被回收,这就需要我们定制__del__,在对象被删除前向操作系统发起关闭数据库链接的系统调用,回收资源。
2、__slots__ (子类不继承,除非子类类定义)
为了达到限制的目的,Python允许在定义class的时候,定义一个特殊的__slots__变量,来限制该class实例能添加的属性:
class Student(object):
__slots__ = ('name', 'age') # 用tuple定义允许绑定的属性名称
3、__module__和__class__
#__module__表示当前操作的对象在哪个模块
#__class__表示当前操作的对象是哪个类的实例化
4.通过type生成类:
# A = type('A',(object,),{'func':func})
# a1 = A()
# a1.func()
A = type('A',(object,),{'func':lambda self:print(666)})
a1 = A()
a1.func()
python全栈开发day23-面向对象高级:反射(getattr、hasattr、setattr、delattr)、__call__、__len__、__str__、__repr__、__hash__、__eq__、isinstance、issubclass的更多相关文章
- Python全栈开发【面向对象进阶】
Python全栈开发[面向对象进阶] 本节内容: isinstance(obj,cls)和issubclass(sub,super) 反射 __setattr__,__delattr__,__geta ...
- Python全栈开发【面向对象】
Python全栈开发[面向对象] 本节内容: 三大编程范式 面向对象设计与面向对象编程 类和对象 静态属性.类方法.静态方法 类组合 继承 多态 封装 三大编程范式 三大编程范式: 1.面向过程编程 ...
- Python 全栈开发七 面向对象
一.编程范式 编程是程序员用特定的语法+数据结构+算法组成的代码来告诉计算机如何执行任务的过程 , 一个程序是程序员为了得到一个任务结果而编写的一组指令的集合,正所谓条条大路通罗马,实现一个任务的方式 ...
- python全栈开发 * 22 面向对象 知识点汇总 * 180703
22 面向对象 -----属性,类方法,静态方法,反射一.属性1.属性的定义:将方法伪装成属性,虽然在代码层面上没有任何高深之处,但让其看起来更合理. (类似于属性的方法)class Person: ...
- python全栈开发 * 23 面向对象 知识点汇总 * 180704
23 面向对象 -----特殊方法 1. isinstance(obj,类名) 判断对象是否是此类实例化或者此类的子类实例化出来的class A:passclass B(A):passb1=B()pr ...
- python全栈开发 * 19 面向对象 知识点汇总 * 180701
19 面向对象初识2 一.类的名称空间1.对象的名称空间对象的名称空间中有类对象指针,对象可以通过"类对象指针"找类中的"静态变量"(静态字段),也可以用对象找 ...
- python全栈开发 * 18 面向对象知识点汇总 * 180530
18 面向对象初识1class person: level="高级动物" mind="有思想" def __init__(self,name,age,gent, ...
- python全栈开发day21面向对象初识总结
- Python笔记【7】_反射getattr&hasattr&setattr&delattr
Lesson0402_GetatrrWebsite.py #!/usr/bin/env/python #-*-coding:utf-8-*- #Author:LingChongShi #查看源码Ctr ...
- python 全栈开发之路 day1
python 全栈开发之路 day1 本节内容 计算机发展介绍 计算机硬件组成 计算机基本原理 计算机 计算机(computer)俗称电脑,是一种用于高速计算的电子计算机器,可以进行数值计算,又可 ...
随机推荐
- vue中$mount与el区别
vue中的$mount和el两者在使用中没有什么区别,都是将实例化后的vue挂载到指定的DOM元素中 用法: 如果在实例化vue时指定el,则该vue将会渲染在el对应的DOM中 反之 没有指定el, ...
- 致敬Python 2.7! 致敬unicode函数!
致敬Python 2.7! 致敬unicode函数! 终于下定决心放弃python 2.7, 拥抱Python 3.x的阵营了. 因为老是被中文编码虐待, 受够了. 同时也把机器里的widows XP ...
- Java——关于num++和++num
public class num_add_add { public static void numAdd(){ int num = 10; int a = num++; System.out.prin ...
- 进度条QProgressBar
import sys from PyQt5.QtCore import Qt, QTimer from PyQt5.QtWidgets import QApplication, QWidget, QP ...
- 挖掘两个Integer对象的swap的内幕
public class SwapTest { public static void main(String[] args) throws Exception { Integer a = 1, b=2 ...
- Eclipse文件路径
经常我们需要读取某个文件,一般情况下,在Eclipse工程中,路径为./src/....
- C#的五种访问修饰符
简述: 所有类型和类型成员都具有可访问性级别,用来控制是否可以在您程序集的其他代码中或其他程序集中使用它们. 可使用访问修饰符指定声明类型或成员的可访问性. 在C#语言中,共有五种访问修饰符:publ ...
- How to Repair GRUB2 When Ubuntu Won’t Boot
Ubuntu and many other Linux distributions use the GRUB2 boot loader. If GRUB2 breaks—for example, if ...
- 【转】深入浅出JMS(二)--ActiveMQ简单介绍以及安装
现实的企业中,对于消息通信的应用一直都非常的火热,而且在J2EE的企业应用中扮演着特殊的角色,所以对于它研究是非常有必要的. 这篇博文介绍一款开源的JMS具体实现——ActiveMQ.ActiveMQ ...
- 【转】CString与string、char*的区别和转换
我们在C++的开发中经常会碰到string.char*以及CString,这三种都表示字符串类型,有很多相似又不同的地方,常常让人混淆.下面详细介绍这三者的区别.联系和转换: 各自的区别 char*: ...