# 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. unix:///var/run/supervisor.sock no such file报错解决办法

    报错 unix:///var/run/supervisor.sock no such file 原因 /var/run/supervisor.sock已被清理. 解决办法 关闭supervisor:p ...

  2. 学python有了这些书你还担心有什么学不会的吗

    百度云盘:Python高级编程PDF高清完整版书籍免费下载 提取码:bn9d 内容简介  · · · · · · <Python高级编程>通过大量的实例,介绍了Python语言的最佳实践和 ...

  3. 上传文件-jq

    var formFile = new FormData(); formFile.append("[file]", fileObj); formFile.append("t ...

  4. ES语法(简)

    1.上编写到怎样安装部署elk,这篇记录怎样简单使用.以便于后期复习使用. 2.登录localhost:5601进入开发者模式. 3.初学担心没有数据可以自己PUT数据比较直观的学习语法.这里可以用网 ...

  5. docker-compose之memcached

    新建docker-compose.yml,写入以下内容 memcached: image: memcached:latest ports: - 11211:11211 启动服务 docker-comp ...

  6. JavaScript 调用Bomb后端云

                                                     用简单的代码 展示代码的魅力 Bmob 是后端云 全方位一体化的后端服务平台 提供可靠的 Server ...

  7. python菜鸟学习: 4.购物车演练

    # -*- coding: utf-8 -*-'''1.启动程序后让用户输入消费额,然后打印商品列表'2.允许用户根据商品编码购买商品3.用户选择商品后,检测余额是否够,够则直接扣款,不够就提醒4.可 ...

  8. 【BOOK】数据存储—文件存储(TXT、JSON、CSV)

    数据存储 文本文件-TXT.JSON.CSV 关系型数据库-MySQL.SQLite.Oracle.SQL Server.DB2 非关系型数据库-MongoDB.Redis   文件打开 open() ...

  9. 学习Vue踩过的坑

    1.用Vue绑定方法的时候里面的methods:要加s!!! 2.v-on只有在事件监听(@click=" 方法名")和不需要参数时候才不要加 ( ),在胡子写法中{{ fun() ...

  10. MySQL事务MVCC、undolog和redolog

    MySql的MVCC多版本控制 undolog:回滚日志(保证一致性)只有在ReadCommited和RepeatableRead隔离级别有用 redolog:重写日志(保证持久性) 示例讲解 Rea ...