Python反射和内置方法(双下方法)
Python反射和内置方法(双下方法)
一、反射
什么是反射
反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。
Python面向对象中的反射
通过字符串的形式操作对象相关的属性。Python中的一切事物都是对象(都可以使用反射)
四个可以实现自省的函数:
hasattr():检测是否含有某属性
class Foo:
def __init__(self, name, age):
self.name = name
self.age = age
def func(self):
pass
# 检测是否含有某属性
print(hasattr(Foo, "func")) # True
f = Foo("dogfa", 18)
print(hasattr(f, "name")) # True
print(hasattr(f, "gender")) # False
getattr():获取属性
class Foo:
def __init__(self, name, age):
self.name = name
self.age = age
def func(self):
pass
@staticmethod
def staticfunc():
print("嘿嘿嘿")
# 获取属性
f = getattr(Foo, "staticfunc")
print(f) # <function Foo.staticfunc at 0x0000028DD8EB9D08>
f() # 嘿嘿嘿
# 如果要获取的属性不存在那么就会报错
f2 = getattr(Foo, "func1") # AttributeError: type object 'Foo' has no attribute 'func1'
setattr():设置属性
class Foo:
def __init__(self, name, age):
self.name = name
self.age = age
def func(self):
pass
@staticmethod
def staticfunc():
print("嘿嘿嘿")
f = Foo("dogfa", 18)
print(f.__dict__) # {'name': 'dogfa', 'age': 18}
setattr(f, "name", f.name + "sb")
print(f.__dict__) # {'name': 'dogfa_sb', 'age': 18}
defattr():删除属性
class Foo:
def __init__(self, name, age):
self.name = name
self.age = age
def func(self):
pass
@staticmethod
def staticfunc():
print("嘿嘿嘿")
f = Foo("dogfa", 18)
print(f.__dict__) # {'name': 'dogfa', 'age': 18}
delattr(f, "name")
print(f.__dict__) # {'age': 18}
# delattr(f, "gender") # 删除不存在的属性时会报错
使用反射的场景
- 对对象的反射
- 对类的反射
- 其它模块的反射
- 当前模块的反射
二、内置方法
函数与方法的区别
在说方法之前我们先来说一下在Python中函数和方法的区别。
# 通过导入types模块来验证
from types import FunctionType, MethodType
def func():
pass
class A:
def func(self):
pass
@staticmethod
def func2():
pass
obj = A()
# FunctionType:函数 MethodType:方法
print(isinstance(func, FunctionType)) # True
print(isinstance(A.func, FunctionType)) # True
print(isinstance(obj.func, FunctionType)) # False
print(isinstance(obj.func, MethodType)) # True
print(isinstance(A.func2, FunctionType)) # True
print(isinstance(obj.func2, FunctionType)) # True
# 通过打印函数(方法)名验证
from types import FunctionType, MethodType
def func():
pass
class A:
def func(self):
pass
@staticmethod
def func2():
pass
obj = A()
print(func) # <function func at 0x000002013BC32E18>
print(A.func) # <function A.func at 0x000002013BF6A598>
print(obj.func) # <bound method A.func of <__main__.A object at 0x000002013BDF7DD8>>
总结:
(1)函数的是显式传递数据的。如我们要指明为len()函数传递一些要处理数据。
(2)函数则跟对象无关。
(3)方法中的数据则是隐式传递的。
(4)方法可以操作类内部的数据。
(5)方法跟对象是关联的。如我们在用strip()方法时,都是要通过str对象调用,比如我们有字符串s,然后s.strip()这样调用。是的,strip()方法属于str对象。
(6)静态方法就是函数
内置方法(双下方法)
定义:双下方法是特殊方法,他是解释器提供的 由爽下划线加方法名加双下划线 __方法名__的具有特殊意义的方法,双下方法主要是Python源码程序员使用的,我们在开发中尽量不要使用双下方法,但是深入研究双下方法,更有益于我们阅读源码。
调用:不同的双下方法有不同的触发方式,就好比盗墓时触发的机关一样,不知不觉就触发了双下方法,例如:__init__
item系列
class Foo:
def __init__(self, name, age):
self.name = name
self.age = age
def __getitem__(self, item):
print("执行obj['key']时会执行我")
return getattr(self, item)
def __setitem__(self, key, value):
print("执行obj['key'] = value 时会执行我")
setattr(self, key, value)
def __delitem__(self, key):
print("执行del['key']时会执行我")
delattr(self, key)
obj = Foo("oldwang", 20)
print(obj["name"])
# 执行obj['key']时会执行我
# oldwang
obj["name"] = "oldniu"
print(obj["name"])
# 执行obj['key'] = value 时会执行我
# 执行obj['key']时会执行我
# oldniu
print(obj.__dict__)
del obj["name"]
print(obj.__dict__)
# {'name': 'oldniu', 'age': 20}
# 执行del['key']时会执行我
# {'age': 20}
__del__
析构方法,当对象在内存中被释放时,自动触发执行。
注:此方法一般无须定义,因为Python是一门高级语言,程序员在使用时无需关心内存的分配和释放,因为此工作都是交给Python解释器来执行,所以,析构函数的调用是由解释器在进行垃圾回收时自动触发执行的。
class FileHandle:
def __init__(self, file_path):
self.f = open(file_path, mode="r", encoding="utf-8")
def read(self):
return self.f.read(1024)
def __del__(self):
# 在程序执行完时释放文件句柄
self.f.close()
f = FileHandle("file/userinfo")
print(f.read())
__new__
# 单例类
class Single:
__isinstance = None
def __new__(cls, *args, **kwargs):
if not cls.__isinstance:
cls.__isinstance = object.__new__(cls)
return cls.__isinstance
return cls.__isinstance
one = Single()
two = Single()
print(one is two) # True
__call__
对象后面加括号,触发执行。
注:构造方法的执行是由创建对象触发的,即:对象 = 类名() ;而对于 call 方法的执行是由对象后加括号触发的,即:对象() 或者 类()()
class Foo:
def __call__(self, *args, **kwargs):
print("执行__call__")
f = Foo()
f() # 执行__call__
# 还可以这样写
Foo()()
__len__
class Foo:
def __init__(self, name, age):
self.name = name
self.age =age
def __len__(self):
return len(self.__dict__)
f = Foo("dogfa", 18)
print(len(f)) # 2
__hash__
class A:
def __init__(self):
self.a = 1
self.b = 2
def __hash__(self):
return hash(str(self.a)+str(self.b))
a = A()
print(hash(a))
__eq__
class A:
def __init__(self):
self.a = 1
self.b = 2
def __eq__(self,obj):
if self.a == obj.a and self.b == obj.b:
return True
a = A()
b = A()
print(a == b) # True
三、其它
# 有一个员工类,1000个员工对象,对象属性有姓名,性别,年龄,部门,
# 按姓名,性别对1000个对象去重
class Employee:
def __init__(self, name, age, gender, department):
self.name = name
self.age = age
self.gender = gender
self.department = department
def __hash__(self):
return hash("{0}{1}".format(self.name, self.gender))
def __eq__(self, other):
if self.name == other.name and self.gender == other.gender:
return True
employee_list = []
for i in range(250):
employee_list.append(Employee("dogfa", i, "male", "python" + str(i)))
for i in range(250):
employee_list.append(Employee("djb", i, "female", "php" + str(i)))
for i in range(250):
employee_list.append(Employee("oldniu", i, "male", "java" + str(i)))
for i in range(250):
employee_list.append(Employee("cdj", i, "female", "go" + str(i)))
for employee_obj in set(employee_list):
print(employee_obj.__dict__)
Python反射和内置方法(双下方法)的更多相关文章
- python 反射和内置方法
一.isinstance和issubclass class Foo: pass class Son(Foo): pass s = Son() #判断一个对象是不是这个类的对象,传两个参数(对象,类) ...
- Python day12部分内置函数的常见方法
#内置函数 print(abs(-1))#绝对值 ']))#判断list的真假,都真则真,有一个假也是假 print(any(''))#有一个真就真,全假则假 print(bin(3))#十进制转换二 ...
- Python面向对象之魔法方法/双下方法
1.__new__ and __init__ 这两个方法都是在实例化的时候执行,__new__ 在 __init__ 之前执行,并且 如果实例化的时候封装属性,__new__也是必须要传的,而且__n ...
- day27:反射和双下方法
1, # 面向对象的三大特性:继承,多态和封装 # 继承: # 单继承: **** # 父类(超类,基类) # 子类(派生类) 派生方法和派生属性 # 子类的对象在调用方法和属性:先用自己的,自己没有 ...
- python面向对象 : 反射和内置方法
一. 反射 1. isinstance()和issubclass() isinstance( 对象名, 类名) : 判断对象所属关系,包括父类 (注:type(对象名) is 类名 : 判断对象所属 ...
- Python面向对象之反射,双下方法
一. 反射 反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力(自省).这一概念的提出很快引发了计算机科学领域关于应用反射性的研究.它首先被程序 ...
- 反射,内置方法,__str__ __repr__
反射 反射用到的mmp模块 def wahaha():print('wahaha') class QQxing: def __init__(self,name): self.name = name d ...
- python之路-双下方法
双下方法 定义: 双下方法是特殊方法,他是解释器提供的,由双下线加方法名加双下划线 __方法名__具有特殊意义的方法 双下方法主要是Python源码程序员使用的,元编程 我们在开发中尽量不要使用双下方 ...
- Python面向对象 | 双下方法
定义:双下方法是特殊方法,他是解释器提供的.由双下划线+方法名+双下划线 .它具有特殊意义的方法,双下方法主要是python源码程序员使用的,我们在开发中尽量不要使用双下方法,但是深入研究双下方法,更 ...
随机推荐
- 【实战】Vue全家桶(vue + axios + vue-router + vuex)搭建移动端H5项目
使用Vue全家桶开发移动端页面. 本博文默认已安装node.js. github链接 一.准备工作 安装vue npm install vue 安装脚手架vue-cli npm install -g ...
- js即时函数在异步回调中的运用
在编程中我们会接触到循环和异步编程的情况,这时异步回调执行逻辑就会出现问题.我们用setTimeout来模拟异步的: for(var i=0;i<3;i++){ setTimeout(funct ...
- Java基础系列 - 接口(功能,用途和优势)
package com.test1; /** * 接口的使用 */ public class test1 { public static void main(String[] args) { //创建 ...
- 【转】Python基础-字符串
原文地址http://blog.chinaunix.net/uid-21169302-id-446256.html Python-String-Function 字符串中字符大小写的变换: * S.l ...
- java打jar包与找不到依赖包详解
eclipse打jar包与找不到依赖包详解 eclipse打工具jar 1.项目右键-->export -->搜索java 2.选择JAR file 3.打包 eclipse打包可执行ja ...
- Python 中路径的有效使用
import arcpy arcpy.GetCount_management("c:/temp/streams.shp") arcpy.GetCount_management(&q ...
- 自定义ObjectAnimator属性实现
package com.loaderman.customviewdemo; import android.animation.ObjectAnimator; import android.graphi ...
- webpack——Modules && Hot Module Replacement
blog:JavaScript Module Systems Showdown: CommonJS vs AMD vs ES2015 官网链接: Modules 官网链接:Hot Module Rep ...
- ElasticSearch——冷热(hot&warm)架构部署
背景 最近在做订单数据存储到ElasticSearch,考虑到数据量比较大,采用冷热架构来存储,每月建立一个新索引,数据先写入到热索引,通过工具将3个月后的索引自动迁移到冷节点上. ElasticSe ...
- Composer更新慢的终极解决方案-转
转自:http://blog.csdn.net/fishermanmax/article/details/51975692 Packagist 镜像 请各位使用本镜像的同学注意: 本镜像已经依照 co ...