python运算符重载(二)
一、基础知识
定义:当类中提供了某个特殊名称的方法,在该类的实例出现在它们相关的表达式时,Python自动调用它们
特性:
1、运算符重载让类拦截常规的Python运算。
2、类可重载所有Python表达式运算符。
3、类可重载打印,函数调用,属性点号运算等内置运算
4、重载使类实例的行为像内置类型
5、重载是通过提供特殊名称的类方法来实现的
二、字典索引和分片:_getitem_,_setitem_,_delitem_
如果在类中定义了(或者继承了)的话,则对于实例中的索引运算,会自动调用__getitem__,__setitem__,__delitem__三个方法。当实例X出现X[i]这样的索引运算时,Python会调用这个实例继承的__getitem__,__setitem__,__delitem__方法(如果有的话)
class Foo:
def __init__(self,name):
self.name=name
def __getitem__(self, item):
return self.__dict__[item]
def __setitem__(self, key, value):
self.__dict__[key]=value
def __delitem__(self, key):
del self.__dict__[key]
F=Foo('egon') #实例化
print(F['name']) #打印,以字典索引的方式,会找到__getitem__方法下的代码,‘name’传递给第二个参数
F['age']=18 #赋值操作,直接传递给__setitem__方法
print(F.__dict__) #打印F的命名空间
del F['age'] #del会调用__delitem__方法,删除‘age’
print(F.__dict__)
打印结果为:
egon
{'name': 'egon', 'age': 18}
{'name': 'egon'}
三、_slots_
__slots__:实例化的对象将没有名称空间,都保存在类的名称空间,而且只能设置指定的属性,例如下面的例子,每个实例只能设置x,y,z三个属性
class People:
__slots__=['x','y','z']
p=People
p.x=1
p.y=2
p.z=3
print(People.__dict__)
p.e=4 #这个将会报错
四、迭代器对象:_iter_,_next_
python 中所有的迭代环境都会先尝试__iter__方法,再尝试__getitem__,也就是说,只有对象在不支持迭代的情况下,才会尝试索引的方式运算。
Python中,迭代环境是通过调用内置函数iter去尝试寻找__iter__方法来实现的,而这种方法返回一个迭代器对象,如果已经提供了,Python就会重复调用这个迭代器对象的next方法,知道发生了StopIteration异常
class Range:
def __init__(self,start,end,long): #构造函数,定义三个元素,start,end,long
self.start=start
self.end=end
self.long=long
def __iter__(self): #__iter__:生成迭代器对象self
return self #返回这个迭代器本身
def __next__(self): #__next__:一个一个返回迭代器内的值
if self.start>=self.end:
raise StopIteration
n=self.start
self.start+=self.long
return n
r=Range(1,10,2) #实例化对象r,
for i in r: #r会首先调用__iter__方法,把自己转换为迭代器
print(i)
五、实现上下文管理:with/as,_enter_,_exit_
with open('a.txt') as f:#with代码块后会自定关闭文件,无论是否发生异常,一下是with的工作方式:
1、计算表达式,所得到的对象称为环境管理器,他必须有__enter__,__exit__方法。
2、环境管理器的__enter__方法会被调用。如果as字句存在,器返回值会赋值给as字句中的变量,否则直接丢弃。
3、代码块中嵌套的代码会执行。
4、如果with代码块引发异常,__exit__(type,value,traceback)方法就会调用(带有异常细节)
5、如果with代码块没有引发异常,__exit__方法依然会被调用,其type,value,traceback参数都会以none传递
import time
class Open:
def __init__(self,filepath,mode='r',encoding='utf8'):
self.filepath=filepath
self.mode=mode
self.encoding=encoding
self.x=open(filepath,mode=mode,encoding=encoding)
def write(self,line):
t=time.strftime('%Y-%m-%d %X')
self.x.write('%s %s' %(t,line))
def __getattr__(self, item):
return getattr(self.x,item)
def __enter__(self):
print("*************")
return self
def __exit__(self, exc_type, exc_val, exc_tb):
print('文件关闭')
self.x.close()
with Open('a.txt','w') as f: #出现with语句,对象的__enter__被触发,有返回值则赋值给as声明的变量
f.write("abc") #调用自己定义的write方法,把‘abc’ 写入到a.txt文件
print('==============') #这一行代码执行完毕后,会触发__exit__方法,关闭文件
print("继续执行代码")
#代码执行如下
*************
==============
文件关闭
继续执行代码
六、_call_:python会为实例用函数调用的表达式运行__call__方法,这样就可以让类实例的外观看起来像是函数
class callee:
def __call__(self, *args, **kwargs):
print('call:',args,kwargs)
C=callee()
C(1,2,3) #实例加()后,会调用__call__方法
C(1,2,3,X=4,Y=5)
#打印结果如下
call: (1, 2, 3) {}
call: (1, 2, 3) {'X': 4, 'Y': 5}
七、_del_:析构函数
每当实例产生时,就对调用__init__构造函数,没当实例空间被收回时,就会调用__del__析构函数。吃方法一般不需定义,python 有自己的内存回收机制
import time
class Open:
def __init__(self,filepath,mode='r',encoding='utf8'):
self.filepath=filepath
self.mode=mode
self.encoding=encoding
self.x=open(filepath,mode=mode,encoding=encoding)
def __del__(self):
print('del')
self.x.close()
f=Open('a.txt','w')
del f
python运算符重载(二)的更多相关文章
- python运算符重载
python运算符重载就是在解释器使用对象内置操作前,拦截该操作,使用自己写的重载方法. 重载方法:__init__为构造函数,__sub__为减法表达式 class Number: def __in ...
- C++ 运算符重载二(一元运算符重载)
//一元运算符重载 #include<iostream> using namespace std; class Point { public: Point(int x,int y){ th ...
- 【C/C++开发】运算符重载二
C++中预定义的运算符的操作对象只能是基本数据类型.但实际上,对于许多用户自定义类型(例如类),也需要类似的运算操作.这时就必须在C++中重新定义这些运算符,赋予已有运算符新的功能,使它能够用于特定类 ...
- Python 运算符重载
https://www.cnblogs.com/hotbaby/p/4913363.html
- python 的重载
python 的重载主要包括方法重载和运算符重载.1.python 方法重载: 其他的语言一般对于方法重载的话,主要是根据参数的类型不同或者是数量不同来区分同名的方法.而python则比较特殊,它本身 ...
- python类可以截获Python运算符
类可以截获Python运算符 现在,让我们来看类和模块的第三个主要差别: 运算符重载.简而言之,运算符重载就是让用类写成的对象,可截获并响应用在内置类型上的运算:加法.切片.打印和点号运算等.这只是自 ...
- Python 正确重载运算符
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px Helvetica } 有些事情让我不安,比如运算符重载.我决定不支持运算符重载,这完全是个人 ...
- Python全栈工程师(异常(高级)、运算符重载)
ParisGabriel 每天坚持手写 一天一篇 决定坚持几年 为了梦想为了信仰 开局一张图 Python人工智能从入门到精通 对象的属性管理函数: getattr ...
- [b0018] python 归纳 (四)_运算符重载
# -*- coding: UTF-8 -*- """ 测试运算符重载 加法 总结: python 运算符表达式其实都是调用 类中方法 __xxx__ + <--- ...
随机推荐
- 关于数据库的左,右,内,外连接,Union和Union all---------笔记
1.左连接 select a.filed1,a.filed2,b.filed1 from a (左表) left join b(右表) on a.commonfiled = b.commonfiled ...
- Java设计模式-责任链模式
提出问题: 最初接触责任链模式就是在struts2中,在当时学的时候看了一眼,大概知道了原理,最近在复习,模拟struts2,说是模拟只是大体模拟了struts2的工作流程,很多东西都是写死的,只是为 ...
- Java爬虫爬取网站电影下载链接
之前有看过一段时间爬虫,了解了爬虫的原理,以及一些实现的方法,本项目完成于半年前,一直放在那里,现在和大家分享出来. 网络爬虫简单的原理就是把程序想象成为一个小虫子,一旦进去了一个大门,这个小虫子就像 ...
- Python中列表、元组、字典增删改查基本区别
1.定义: 列表:num = ["a","b"."c"] ##定义后可增删改查 元组:num = ("a"," ...
- Jenkins集成Docker镜像实现自动发布
1. 思路&流程 Jenkins集成Docker镜像实现自动发布与Jenkins发布mavne项目思路一样总体流程 为:Jenkins 拉去远端源码 -- gitl实现应用打包 -- jenk ...
- 数据库(Mongodb)
1.MongoClient()函数 In [8]: import pymongo In [9]: con = pymongo.MongoClient('localhost') #建立连接 In [10 ...
- vs运行单个cpp文件
打开vs,新建项目,左侧win32见上图,右侧 win32控制台应用程序,填好名称后,确定----下一步,如下图,空项目 紧接着如下图,通过现有项添加自己的cpp文件,便可以运行了
- Jquery给网页的title取值和赋值
//获取title的值 var title_val=$('#id').attr('title'); alert(title_val); //修改title的值.赋值给title $('#id2').a ...
- Git详细教程(2)---多人协作开发
Git可以完成两件事情: 1. 版本控制 2.多人协作开发 如今的项目,规模越来越大,功能越来越多,需要有一个团队进行开发. 如果有多个开发人员共同开发一个项目,如何进行协作的呢. Git提供了一个非 ...
- Spark ML源码分析之四 树
之前我们讲过,在Spark ML中所有的机器学习模型都是以参数作为划分的,树相关的参数定义在treeParams.scala这个文件中,这里构建一个关于树的体系结构.首先,以Decis ...