Python之路(第二十九篇) 面向对象进阶:内置方法补充、异常处理

时间:2021-07-07 22:32:01

一、__new__方法

__init__()是初始化方法,__new__()方法是构造方法,创建一个新的对象

实例化对象的时候,调用__init__()初始化之前,先调用了__new__()方法

__new__()必须要有返回值,返回实例化出来的实例

  def __new__(cls, *args, **kwargs):

例子

  class Foo:

def __init__(self,name):
self.name = name

def __new__(cls, *args, **kwargs):
obj = object.__new__(Foo) #调用object创建新的空对象
obj.__init__(*args, **kwargs) #调用对象的__init__方法对对象属性字典增加属性
return obj #返回新创建的属性给f

f = Foo("nick") #这里直接调用__call__方法
print(f.name)
print(f.__dict__)

  

二、__len__方法

如果一个类表现得像一个list,要获取有多少个元素,就得用 __len__() 函数。

例子

  
  class Student():
def __init__(self,*args):
self.names = args

def __len__(self):
return len(self.names)

s = Student("nick","jack","pony")
print(len(s)) #用len直接调用__len__方法

  

三、__eq__方法

__eq__(slef,other) ,判断self对象是否等于other对象,使用==或者is调用此方法。

例子

  class Foo:

def __init__(self,name):
self.name = name

def __eq__(self, other):
if self.name == other.name: #判断如果对象的name属性相等就返回True
return True
else:
return False

obj1 = Foo("nick")
obj2 = Foo("nicholas")
print(obj1 is obj2)

  

四、__hash__方法

获取取一个对象的哈希值,一般来说,对于对象的hash值是基于对象的内存地址生成的,但是重写__hash__方法可以自己定制hash取值的对象

例子

  class Foo:

def __init__(self,name,age):
self.name = name
self.age = age

def __hash__(self):
return hash(self.name+str(self.age)) #这里基于对象的两个属性返回hash值


obj1 = Foo("nick",18) #注意hash的对象不能是整数
obj2 = Foo("nick",18)
print(hash(obj1))
print(hash(obj2))

  

五、经典代码分析

例子1、纸牌游戏

  from collections import namedtuple

Card = namedtuple("Card",["rank","suit"]) #这里相当于创建了一个名为Card的类,后面的列表里是类的类属性,没有函数属性
#
class FranchDeck:
ranks = [str(n) for n in range(2,11)] + list('JQKA')
suits = ['红心','方板','梅花','黑桃']

def __init__(self):
self._cards = [Card(rank,suit) for rank in FranchDeck.ranks for suit in FranchDeck.suits] #这里是一个列表生成式,
#将第一个for循环里的值赋值给rank,将第二个for循环里的值赋给suit,这个类只要一实例化就自动生成了一副除了没有大小王的牌
#每张牌都是用元组来表示的
def __len__(self):
return len(self._cards) # choice()方法依赖__len__(self)

def __getitem__(self, item):
return self._cards[item]

def __setitem__(self, key, value):
self._cards[key] = value

deck = FranchDeck() #这里相当于创建了一副牌
print(deck[0]) #打印第一张牌
print(deck[2:5]) #打印第2-4张牌
from random import choice,shuffle
print(choice(deck)) #随机抽取一张牌
shuffle(deck) #洗牌
print("洗牌后的第一张",deck[0]) #打印第一张牌

# 如果不实现,__getitem__(self, item)和__setitem__(self, key, value)
# 会报错:TypeError: 'FranchDeck' object does not support indexing,不支持下标索引,
# 因为这里的随机洗牌,会用到下标方式取牌和放牌.

  

例子2

去重,去除对象中名字、性别相同、但年龄不同的对象

这里要用到set()函数,如果set函数参数为空,则返回一个空的set对象 如果参数是一个可迭代对象时,则返回去重的一个set对象。

set()函数去重需要用到__eq__()、__hash__()内置方法。

  
  class People:

def __init__(self,name,sex,age):
self.name = name
self.sex = sex
self.age = age

def __eq__(self, other):
if self.name == other.name and self.sex == other.sex:
return True
return False

def __hash__(self):
return hash(self.name+self.sex)

obj1 = People("nick","male",18)
obj2 = People("nick","male",20)
print(set((obj1,obj2))) #注意set()函数传入的参数必须是可迭代对象,这里用的是元组
li = [obj1,obj2]
print(set(li)) #也可以用列表

  

六、异常处理

1、什么是异常

异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止)

错误分成两种:语法错误和逻辑错误

语法错误是一些代码的标点符号错误等。逻辑错误有很多种。

语法错误示例:

  
  #语法错误示范一
if
#语法错误示范二
def test:
pass
#语法错误示范三
class Foo
pass
#语法错误示范四
print(haha

1.语法错误(这种错误过不了python解释器的语法检测,必须在程序执行前就改正)

  

逻辑错误示例

  
  #TypeError:int类型不可迭代
for i in 3:
pass #ValueError
num=input(">>: ") #输入hello
int(num)

#NameError
aaa

#IndexError
l=['egon','aa']
l[3]

#KeyError
dic={'name':'egon'}
dic['age']

#AttributeError
class Foo:pass
Foo.x

#ZeroDivisionError:无法完成计算
res1=1/0
res2=1+'str'

  

2、异常的种类

常用异常

  AttributeError 试图访问一个对象没有的属性,比如foo.x,但是foo没有属性x
IOError 输入/输出异常;基本上是无法打开文件
ImportError 无法引入模块或包;基本上是路径问题或名称错误
IndentationError 语法错误(的子类) ;代码没有正确对齐
IndexError 下标索引超出序列边界,比如当x只有三个元素,却试图访问x[5]
KeyError 试图访问字典里不存在的键
KeyboardInterrupt Ctrl+C被按下
NameError 使用一个还未被赋予对象的变量
SyntaxError Python代码非法,代码不能编译,语法错误
TypeError 传入对象类型与要求的不符合
UnboundLocalError 试图访问一个还未被设置的局部变量,基本上是由于另有一个同名的全局变量,
导致你以为正在访问它
ValueError 传入一个调用者不期望的值,即使值的类型是正确的

  

其他异常

  异常名称    描述
BaseException 所有异常的基类
SystemExit 解释器请求退出
KeyboardInterrupt 用户中断执行(通常是输入^C)
Exception 常规错误的基类
StopIteration 迭代器没有更多的值
GeneratorExit 生成器(generator)发生异常来通知退出
StandardError 所有的内建标准异常的基类
ArithmeticError 所有数值计算错误的基类
FloatingPointError 浮点计算错误
OverflowError 数值运算超出最大限制
ZeroDivisionError 除(或取模)零 (所有数据类型)
AssertionError 断言语句失败
AttributeError 对象没有这个属性
EOFError 没有内建输入,到达EOF 标记
EnvironmentError 操作系统错误的基类
IOError 输入/输出操作失败
OSError 操作系统错误
WindowsError 系统调用失败
ImportError 导入模块/对象失败
LookupError 无效数据查询的基类
IndexError 序列中没有此索引(index)
KeyError 映射中没有这个键
MemoryError 内存溢出错误(对于Python 解释器不是致命的)
NameError 未声明/初始化对象 (没有属性)
UnboundLocalError 访问未初始化的本地变量
ReferenceError 弱引用(Weak reference)试图访问已经垃圾回收了的对象
RuntimeError 一般的运行时错误
NotImplementedError 尚未实现的方法
SyntaxError Python 语法错误
IndentationError 缩进错误
TabError Tab 和空格混用
SystemError 一般的解释器系统错误
TypeError 对类型无效的操作
ValueError 传入无效的参数
UnicodeError Unicode 相关的错误
UnicodeDecodeError Unicode 解码时的错误
UnicodeEncodeError Unicode 编码时错误
UnicodeTranslateError Unicode 转换时错误
Warning 警告的基类
DeprecationWarning 关于被弃用的特征的警告
FutureWarning 关于构造将来语义会有改变的警告
OverflowWarning 旧的关于自动提升为长整型(long)的警告
PendingDeprecationWarning 关于特性将会被废弃的警告
RuntimeWarning 可疑的运行时行为(runtime behavior)的警告
SyntaxWarning 可疑的语法的警告
UserWarning 用户代码生成的警告

  

3、异常处理

python解释器检测到错误,触发异常(也允许程序员自己触发异常)

程序员编写特定的代码,专门用来捕捉这个异常(这段代码与程序逻辑无关,与异常处理有关)

如果捕捉成功则进入另外一个处理分支,执行你为其定制的逻辑,使程序不会崩溃,这就是异常处理

4、异常处理的意义

python解析器去执行程序,检测到了一个错误时,触发异常,异常触发后且没被处理的情况下,程序就在当前异常处终止,后面的代码不会运行,所以你必须提供一种异常处理机制来增强你程序的健壮性与容错性。

5、如何进行异常处理

如果错误发生的条件是可预知的,我们需要用if进行处理:在错误发生之前进行预防

  AGE=10
while True:
age=input("请输入年龄: ").strip()
if age.isdigit(): #只有在age为字符串形式的整数时,下列代码才不会出错,该条件是可预知的
age=int(age)
if age == AGE:
print('you got it')
break

  

如果错误发生的条件是不可预知的,则需要用到try...except:在错误发生之后进行处理

try...except

  基本语法为
try:
  被检测的代码块
except 异常类型:
  try中一旦检测到异常,就执行这个位置的逻辑

 #举例
try:
f=open('a.txt')
g=(line.strip() for line in f)
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g))
except StopIteration:
f.close()

  

6、try...except的详细用法

.异常类只能用来处理指定的异常情况,如果非指定异常则无法处理

  s1 = 'hello'
try:
int(s1)
except IndexError as e: # 未捕获到异常,程序直接报错
print e

  

多分支与万能异常

多分支

  s1 = 'hello'
try:
int(s1) #这里产生那种错误就执行那个类型错误下的处理方式
except IndexError as e:
print("IndexError")
print(e)
except KeyError as e:
print("KeyError")
print(e)
except ValueError as e:
print("ValueError")
print(e)

  

万能异常

万能异常是Exception,可以捕获任何异常。

  s1 = 'hello'
li = [1,2,3]
try:
int(s1)
print(li[4])
except Exception as e:
print(e)
如果你想要的效果是,无论出现什么异常

  

,我们统一丢弃,或者使用同一段代码逻辑去处理他们,那么直接用万能异常Exception处理就行。

如果你想要的效果是,对于不同的异常我们需要定制不同的处理逻辑,那就需要用到多分支了。

也可以在多分支的最后加一个Exception。

try...except...else

  s1 = 'hello'
try:
int(s1)
except IndexError as e:
print(e)
except KeyError as e:
print(e)
except ValueError as e:
print(e)
#except Exception as e:
# print(e)
else:
print('try内代码块没有异常则执行我') #没有报异常就执行这个,与for..else类似
finally:
print('无论异常与否,都会执行该模块,通常是进行清理工作') #收尾的finally

  

主动触发异常

raise 主动报出异常

  raise [Exception [, args [, traceback]]]

语句中Exception是异常的类型(例如,NameError)参数是一个异常参数值。该参数是可选的,如果不提供,异常的参数是"None"。

最后一个参数是可选的(在实践中很少使用),如果存在,是跟踪异常对象。主动触发的异常也可以被捕捉到。

  try:
raise TypeError('类型错误')
except Exception as e:
print(e)

  

自定义异常

自定义异常的两种方式

  # 方式一
class MYERROR(Exception):

def __init__(self):
self.err = "自定义异常"

def __str__(self):
return self.err

e = MYERROR()
raise MYERROR

#方式二
class MYERROR2(BaseException):

def __init__(self,msg):
self.msg = msg
#继承BaseException类不再需要__str__方法,BaseException中已经实现了
try:
raise MYERROR2("自定义异常")
except Exception as e:
print("-----")
print(e)

  

      

python的异常继承树

Python之路(第二十九篇) 面向对象进阶:内置方法补充、异常处理

Python之路(第二十九篇) 面向对象进阶:内置方法补充、异常处理

7、assert

assert是断言的意思,我断定这个程序执行之后或者之前会有这样的结果,如果不是,那就扔出一个错误。

语法:

  assert expression [, arguments] 
assert 表达式 [, 参数]

arguments是断言的值

例子

  li = [1,2,3,4]
assert len(li) >= 5, '列表元素个数大于5'

assert 1 != 1,"断言"

  

8、什么时候用异常处理

try...except应该尽量少用,因为给你的程序加了一种异常处理的逻辑,会导致代码可读性变差。

而且异常处理,只有在有些异常无法预知的情况下,才加上try...except,其他的逻辑错误应该尽量修正。

Python之路(第二十九篇) 面向对象进阶:内置方法补充、异常处理的更多相关文章

  1. Python之路(第二十六篇) 面向对象进阶:内置方法

    一.__getattribute__ object.__getattribute__(self, name) 无条件被调用,通过实例访问属性.如果class中定义了__getattr__(),则__g ...

  2. Python之路(第二十八篇) 面向对象进阶:类的装饰器、元类

    一.类的装饰器 类作为一个对象,也可以被装饰. 例子 def wrap(obj): print("装饰器-----") obj.x = 1 obj.y = 3 obj.z = 5 ...

  3. Python之路(第二十五篇) 面向对象初级:反射、内置方法

    [TOC] 一.反射 反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力(自省).这一概念的提出很快引发了计算机科学领域关于应用反射性的研究.它 ...

  4. Python之路(第二十四篇) 面向对象初级:多态、封装

    一.多态 多态 多态:一类事物有多种形态,同一种事物的多种形态,动物分为鸡类,猪类.狗类 例子 import abc class H2o(metaclass=abc.ABCMeta): ​ def _ ...

  5. Python之路(第二十二篇) 面向对象初级:概念、类属性

    一.面向对象概念 1. "面向对象(OOP)"是什么? 简单点说,“面向对象”是一种编程范式,而编程范式是按照不同的编程特点总结出来的编程方式.俗话说,条条大路通罗马,也就说我们使 ...

  6. Python之路(第十九篇)hashlib模块

    一.hashlib模块 HASH Hash,一般翻译做“散列”,也有直接音译为”哈希”的,就是把任意长度的输入(又叫做预映射,pre-image),通过散列算法,变换成固定长度的输出,该输出就是散列值 ...

  7. Python开发【第二十二篇】:Web框架之Django【进阶】

    Python开发[第二十二篇]:Web框架之Django[进阶]   猛击这里:http://www.cnblogs.com/wupeiqi/articles/5246483.html 博客园 首页 ...

  8. python中有两个下划线__的是内置方法,一个下划线_或者没有下划线的可能是属性,也可能是方法,也可能是类名

    python中有两个下划线__的是内置方法,一个下划线_或者没有下划线的可能是属性,也可能是方法,也可能是类名,如果在类中定义的就是类的私有成员. >>> dir(__builtin ...

  9. python 面向对象进阶之内置方法

    一 isinstance(obj,cls)和issubclass(sub,super) 1.1,isinstance(obj,cls)检查是否obj是否是类 cls 的对象 class Foo(obj ...

随机推荐

  1. jquery格式化时间戳 2011-01-01

    /*                * 时间戳转换日期                * @param  <int>  unixTime    待时间戳(秒)               ...

  2. VS2008 工程中部分文件不参与编译 从生成中排除【Worldsing笔记】

    Visual Studio 2008 .VS2008.VC2008工程源文件配置.编译配置   有时编写代码时,往往存在这样的需求(或是希望有这样的功能):一个工程经过不共同的配置实现不同的版本或是功 ...

  3. AVD设置屏幕大小

    相关资料: 1.http://jingyan.baidu.com/article/fedf0737775d2835ac897700.html

  4. 水题:HDU 5119 Happy Matt Friends

    Matt has N friends. They are playing a game together.Each of Matt's friends has a magic number. In t ...

  5. 尽量使用ToUpper比较,避免使用ToLower

    在编码时尽量使用ToUpper比较,避免使用ToLower,因为微软对ToUpper进行了优化,以下为测试结果: public void TestToLower() { Stopwatch watch ...

  6. 转:PHP的&lpar;Thread Safe与Non Thread Safe&rpar;

    在安装xdebug到时候你会有有TS和NTS版本的选择,在以前还有VC6和VC9的版本.如果你没有根据你目前的服务器的状况选择对应的版本的话,那么xdebug是安装不成功的. 一.如何选择 php5. ...

  7. Spark mllib多层分类感知器在情感分析中的实际应用

    import org.apache.spark.ml.Pipeline import org.apache.spark.ml.classification.MultilayerPerceptronCl ...

  8. 托布利兹变换 toeplitz 变换

    托布利兹变换 toeplitz 变换 算术平均变换 '''An->C=>(A1+A2+A3+...+An)/n->C,K_ni=1/n=>+[AiK_ni->C (Yn- ...

  9. package&period;json安装依赖的箭头?

  10. 【黑金原创教程】【Modelsim】【第五章】仿真就是人生

    声明:本文为黑金动力社区(http://www.heijin.org)原创教程,如需转载请注明出处,谢谢! 黑金动力社区2013年原创教程连载计划: http://www.cnblogs.com/al ...