python - 无法腌制日期时间子类
问题描述
我正在尝试腌制和取消腌制datetime.datetime
子类对象。但是,它总是会产生错误,我不知道为什么会这样以及如何解决它。这是最小的示例:
from datetime import datetime, date, time
class A(datetime):
def __new__(cls, year = 2016, month=1, day=1, hour=0, minute=0, leap_year=False):
return datetime.__new__(cls, year ,month, day, hour, minute)
import pickle
obj = A( month=1, day=1, hour=1, leap_year = False)
serial = pickle.dumps(obj, protocol=pickle.HIGHEST_PROTOCOL)
unpickle = pickle.loads( serial, encoding='bytes')
这将给出以下错误:
TypeError Traceback (most recent call last)
<ipython-input-2-12195a09d83d> in <module>()
5
6
----> 7 unpickle = pickle.loads( serial, encoding='bytes')
<ipython-input-1-605483566b52> in __new__(cls, year, month, day, hour, minute, leap_year)
2 class A(datetime):
3 def __new__(cls, year = 2016, month=1, day=1, hour=0, minute=0, leap_year=False):
----> 4 return datetime.__new__(cls, year ,month, day, hour, minute)
5
TypeError: an integer is required (got type bytes)
有谁知道问题可能出在哪里以及如何解决?
解决方案
根据模块文档的Pickling Class Instances部分中的信息,我能够完成以下工作。从已添加的方法返回的值的元组将导致在调用取消调用类实例时调用构造函数——就像调用类时通常发生的情况一样。pickle
__reduce_ex__()
__new__()
pickle.loads()
请注意,我不需要知道是否或如何datetime
自定义酸洗,也不需要了解它的 C 实现。
另请注意,由于leap_year
您的问题中的实现忽略了您的参数(并且尚不清楚为什么无论如何都需要将其传递给初始化程序),因此我已将其替换为 Pythonproperty
动态计算布尔值基于当前实例的年份值。
from datetime import datetime, date, time
import pickle
class A(datetime):
def __new__(cls, year=2016, month=1, day=1, hour=0, minute=0):
return datetime.__new__(cls, year, month, day, hour, minute)
def __reduce_ex__(self, protocol):
return (type(self), (self.year, self.month, self.day, self.hour, self.minute))
@property
def is_leapyear(self):
''' Determine if specified year is leap year. '''
year = self.year
if year % 4 != 0:
return False
elif year % 100 != 0:
return True
elif year % 400 != 0:
return False
else:
return True
obj = A(month=1, day=1, hour=1, leap_year=False)
print('calling pickle.dumps()')
serial = pickle.dumps(obj, protocol=pickle.HIGHEST_PROTOCOL)
print('calling pickle.loads()')
unpickle = pickle.loads(serial, encoding='bytes')
print('unpickled {!r}'.format(unpickle)) # -> unpickled A(2016, 1, 1, 1, 0)
print('unpickle.is_leapyear: {}'.format(unpickle.is_leapyear)) # -> True
推荐阅读
- javascript - 在日期选择器中显示今天的日期
- android - 如何比较 Firebase 中的子值?
- reactjs - 为 react 安装 react-redux@5.0.7 时出错
- html - Bootstrap 4 数据表CSS修改?
- python - Winsound减慢python程序
- telegram-bot - 如何在使用 Bots.Business 创建的机器人中单击链接后检查网站访问情况
- javascript - 如何在Vue的JS文件中导入文件变量?
- python - CTC 损失计算不正确
- python - 使用 tabula.py 从 PDF 格式读取没有标题的表格
- python - 点集并根据发生情况确定它们