# encoding: utf-8
import logging

ERROR_NOT_FOUNDED_FILE = "error_not_founded_file"

class GeneralError(Exception):

def __init__(self, value, description):
print("GeneralError __init__")
self.value = value
self.description = description

def __repr__(self):
print("__repr__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description)

def __str__(self):
print("__str__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description)

class FileError(GeneralError):

def __init__(self, description):
super(FileError, self).__init__(ERROR_NOT_FOUNDED_FILE, description)

def __repr__(self):
return super(FileError, self).__repr__() # 不规范

def __str__(self):
super(FileError, self).__str__() # 会报错

try:
raise FileError('file error')
except Exception as e:
logging.info(f"exec error: {e}")
print("补救措施!")

 执行结果如下:

错误:
__str__ returned non-string (type NoneType) 可以看出,e返回了None。

debug:

    def __str__(self):
super(FileError, self).__str__() # 调用了,父类的__str__函数,没有返回(也可以理解为返回了None)。
logging.info(f"exec error: {e}")  # 调用了FileError().__str__,没有返回值,自然就报错了。

修复:

def __str__(self):
return super(FileError, self).__str__() # 调用了,父类的__str__函数,返回执行结果。

或者
不实现重写__str__。

那么正常的代码应该是:

# encoding: utf-8

import logging

ERROR_NOT_FOUNDED_FILE = "error_not_founded_file"

class GeneralError(Exception):

def __new__(cls, *args, **kwargs):
print("GeneralError __new__")
return super(GeneralError, cls).__new__(cls)

def __init__(self, value, description):
print("GeneralError __init__")
self.value = value
self.description = description

def __repr__(self):
print("__repr__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description)

def __str__(self):
print("__str__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description)

class FileError(GeneralError):

def __init__(self, description):
super(FileError, self).__init__(ERROR_NOT_FOUNDED_FILE, description)

def __repr__(self):
return super(FileError, self).__repr__()

  # 我选择不重写!!! 唯一的修改
# def __str__(self):
# super(FileError, self).__str__()

try:
raise FileError('file error')
except Exception as e:
logging.info(f"exec error: {e}")
print("补救措施!")

当然,这个环境里,代码能跑了。不过,能跑是能跑,有一处不规范

def __repr__(self):
print("__repr__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description)

官方文档对于__repr__的解释是(repr会调用对象的__repr__):

def repr(obj): # real signature unknown; restored from __doc__
"""
Return the canonical string representation of the object. For many object types, including most builtins, eval(repr(obj)) == obj.
"""
pass

很显然:

err = FileError("file  error")
eval(repr(err)) == err File "E:/coding/hello.py", line 53, in <module>
eval(repr(err)) == err
File "<string>", line 1
<FileError error_not_founded_file:file error>
^
SyntaxError: invalid syntax

所以,要修改代码为:

    def __repr__(self):
print("__repr__")
return "{}('{}')".format(self.__class__.__name__, self.description)

那么,最终的代码应该是:

# encoding: utf-8

import logging

ERROR_NOT_FOUNDED_FILE = "error_not_founded_file"

class GeneralError(Exception):

    def __new__(cls, *args, **kwargs):
print("GeneralError __new__")
return super(GeneralError, cls).__new__(cls) def __init__(self, value, description):
print("GeneralError __init__")
self.value = value
self.description = description def __repr__(self):
print("__repr__")
return "{}('{}')".format(self.__class__.__name__, self.description) # 第一处修改: 返回字符串格式修改 def __str__(self):
print("__str__")
return "<{} {}:{}>".format(self.__class__.__name__, self.value, self.description) class FileError(GeneralError): def __init__(self, description):
super(FileError, self).__init__(ERROR_NOT_FOUNDED_FILE, description)
  
# 第二处修改:不重写父类方法

# def __repr__(self):
# return super(FileError, self).__repr__() # def __str__(self):
# super(FileError, self).__str__() try:
raise FileError('file error')
except Exception as e:
logging.info(f"exec error: {e}")
print("补救措施!") err = FileError("file error")
eval(repr(err)) == err

python-魔法函数-__str__ __repr__ 的一次异常的更多相关文章

  1. python魔法函数__dict__和__getattr__的妙用

    python魔法函数__dict__和__getattr__的妙用 __dict__ __dict__是用来存储对象属性的一个字典,其键为属性名,值为属性的值. 既然__dict__是个字典那么我们就 ...

  2. Python魔法函数与两比特量子系统模拟

    技术背景 本文主要涵盖两个领域的知识点:python的魔法函数和量子计算模拟,我们可以通过一个实际的案例来先审视一下这两个需求是如何被结合起来的. 量子计算模拟背景 ProjectQ是一个非常优雅的开 ...

  3. 复习python的__call__ __str__ __repr__ __getattr__函数 整理

    class Www: def __init__(self,name): self.name=name def __str__(self): return '名称 %s'%self.name #__re ...

  4. Python魔法函数

    python中定义的以__开头和结尾的的函数.可以随意定制类的特性.魔法函数定义好之后一般不需要我们自己去调用,而是解释器会自动帮我们调用. __getitem__(self, item) 将类编程一 ...

  5. python基础---- __getattribute__----__str__,__repr__,__format__----__doc__----__module__和__class__

    目录: 一. __getattribute__ 二.__str__,__repr__,__format__ 三.__doc__ 四.__module__和__class__ 一. __getattri ...

  6. python魔法函数之__getattr__与__getattribute__

    getattr 在访问对象的属性不存在时,调用__getattr__,如果没有定义该魔法函数会报错 class Test: def __init__(self, name, age): self.na ...

  7. python魔法函数(二)之__getitem__、__len__、__iter__

    魔法函数会增强python类的类型,独立存在 __getitem class Company: def __init__(self, employees): self.employees = empl ...

  8. python魔法函数的一些疑问

    看了魔法函数,有一点疑问.1中需要用self.word才能执行,而2直接用self就可以执行.而1中Word继承了int基本类型,但在__new__时并没有什么卵用.当用 Word(“123”)来实例 ...

  9. raindi python魔法函数(一)之__repr__与__str__

    __repr__和__str__都是python中的特殊方法,都是用来输出实例对象的,如果没有定义这两个方法在打印的时候只会输出实例所在的内存地址 这种方式的输出没有可读性,并不能直观的体现实例.py ...

  10. Python——详解__str__, __repr__和__format__

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是Python专题的第10篇文章,我们来聊聊Python当中的类. 打印实例 我们先从类和对象当中最简单的打印输出开始讲起,打印一个实例 ...

随机推荐

  1. Ubuntu20.04上用tmux管理新进程

    sudo apt-get install tmux 安装tmux tmux new -s session_name 新开一个会话 tmux a -t session_name 查看指定会话 tmux ...

  2. ubuntu 20.04 更新镜像源

    首先备份系统自带的源,以防修改错误! 命令: sudo cp /etc/apt/sources.list /etc/apt/sources.list.back 1.修改系统镜像源 文件目录:/etc/ ...

  3. iOS开发之定时器和tableview滑动阻塞问题

    NSTimer *timer = [[NSTimer alloc]initWithFireDate:[NSDate distantPast] interval:1 target:self select ...

  4. Docker学习——Dockerfile 指令详解(五)

    我们已经介绍了 FROM (指定基础镜像) , RUN(执行命令) ,还提及了 COPY , ADD ,其实 Dockerfile 功能很强大,它提供了十多个指令.下面我们继续讲解其他的指令. COP ...

  5. eclipse中同步git代码报错checkout conflict with files

    1.Team--->Synchronize Workspace 2.在同步窗口找到冲突文件,把自己本地修改的复制出来 3.在文件上右键选择 Overwrite----->Yes , 4.再 ...

  6. Linux 使用Postfix与Dovecot部署邮件系统

    电子邮件系统 电子邮件系统基于邮件协议来完成电子邮件的传输,常见的邮件协议有下面这些. 简单邮件传输协议(Simple Mail Transfer Protocol,SMTP):用于发送和中转发出的电 ...

  7. AIO基本编写

    一. public class Server { //线程池 private ExecutorService executorService; //线程组 private AsynchronousCh ...

  8. python基础篇 26-redis操作

    redis的基本操作: redis_conf ={ 'host':'192.168.64.128', 'password':'Aa123456', 'db':'0', 'port':6379, 'de ...

  9. Jmeter、Postman之RSA加密登录接口测试

    方法1:直接用在线加密工具进行加密,得到密码 参考地址 https://www.toolscat.com/decode/rsa 输入公钥和密码,直接加密即可   方法2:postman工具 步骤1:接 ...

  10. github秒变vscode

    方法一:github.sth -> github1s.sth 方法二:github保持登录状态,然后是英文输入法,点击'.'就行