python attributeerror nonetype_Python自动析构时出现Exception AttributeError: 'NoneType' object has no attri...

在将print替换为logging后,作者遇到'NoneType' object has no attribute错误。问题源于Python对象析构时尝试访问已销毁的logging对象。解决方案是在类中添加一个关闭数据库连接的close方法,并在主函数结束前调用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

昨晚在整理自己的 python 脚本的时候,想把其中一个脚本中的 print 函数全都改成 logging 包中的相关函数。改完后一运行却出现了 Exception AttributeError: 'NoneType' object has no attribute 的错误,网上搜了一下没找到相关答案。上午再想了想,原因应该是跟python对象的析构有关,具体分析过程如下:

1 示例程序

由于原脚本业务部分过长,这里把关键的几个部位抽出来做了个演示程序。

# -*- coding: UTF-8 -*-

# File: destrution_attribute_error_nonetype1.py

# Description: python自动析构时出现Exception AttributeError: 'NoneType' object has no attribute问题的示例程序

# (c) 2018.12.19 vfhky https://typecodes.com/python/destrution_attribute_error_nonetype1.html

import threading

import logging

# MYSQL 的简单封装

class CMySQL:

# 线程锁

_instance_lock = threading.Lock()

# 数据库连接对象

__db = None

# 游标对象

__cursor = None

def __init__(self, *args, **kwargs):

pass

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

pass

# 析构函数,释放对象时使用

def __del__(self):

# 关闭 数据库 连接

if self.__db:

self.__db.close()

# 自动析构时这里会出问题:'NoneType' object

logging.info("-------> close db.")

else:

# 自动析构时这里也会出问题:'NoneType' object

logging.warning("-------> db unconnected or had been closed.")

if __name__ == "__main__":

# 创建实例

db_obj = CMySQL()

# 输出到控制台

logging.basicConfig(level=logging.INFO,

format='[%(asctime)s][L:%(lineno)d][%(levelname)s][%(process)d] %(message)s',

datefmt='%d %b %Y %H:%M:%S')

logging.info("================================== END ==================================")

2 执行后出现错误

执行上面的程序,在 Linux 终端上就会出现 Exception AttributeError: 'NoneType' object has no attribute 'warning'" in > 的错误。

如下图所示:

cd3656603120b6633525d8a613ff1438.png

3 分析问题

其实是不了解python的析构过程导致的:当main函数结束后(输出图中的 END 字样),意味着进程即将退出,那么会自动调用对象的析构函数进行析构,这点Python和C++是一样的。

由于 logging 模块中的类对象(包括成员变量、成员函数等)已经被析构了,所以当执行 CMySQL 对象的析构函数 __del__ 中的 logging.warning 函数时会出现 "'NoneType' object has no attribute 'warning' 的错误。

4 解决问题

解决方法很简单,只要增加一个封装 MySQL 链接关闭的函数 close 就行了,当main函数结果调用即可。下面的代码是针对这个问题的改进版本。

# -*- coding: UTF-8 -*-

# File: destrution_attribute_error_nonetype1_1.py

# Description: 修正Exception AttributeError: 'NoneType' object has no attribute问题的示例程序

# (c) 2018.12.19 vfhky https://typecodes.com/python/destrution_attribute_error_nonetype1.html

import threading

import logging

# MYSQL 的简单封装

class CMySQL:

# 线程锁

_instance_lock = threading.Lock()

# 数据库连接对象

__db = None

# 游标对象

__cursor = None

def __init__(self, *args, **kwargs):

pass

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

pass

# 关闭数据库连接

def close(self):

if self.__db:

self.__db.close()

self.__db = None

self.__cursor = None

logging.info("-------> close db.")

else:

logging.warning("-------> db unconnected or had been closed.")

# 析构函数,释放对象时使用

def __del__(self):

# 关闭数据库连接

if self.__db:

self.__db.close()

self.__db = None

self.__cursor = None

# logging.info("-------> close db.")

else:

pass

#logging.warning("-------> db unconnected or had been closed.")

if __name__ == "__main__":

#

db_obj = CMySQL()

# 输出到控制台

logging.basicConfig(level=logging.INFO,

format='[%(asctime)s][L:%(lineno)d][%(levelname)s][%(process)d] %(message)s',

datefmt='%d %b %Y %H:%M:%S')

logging.info("================================== END ==================================")

# 关闭数据库连接

db_obj.close()

运行效果如下图所示:

1f862d984182d738034324f1ef2e595e.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值