如何解决深度复制不会产生唯一的对象
class loginfo:
info = {
'global_time': -1,'exptime': 0,'type': ""
# omitted other entries for brevity
}
def assign(self,d):
if d is not None:
self.info.update((key,d[key]) for key in d if key in self.info)
def __str__(self):
s = str(self.info['global_time'])
for key in self.info:
if key == 'global_time':
continue
s += ",\t"
s += str(self.info[key])
s += "\n"
return(s)
我从有点奇怪的日志文件中读取行,并在单独的例程parse_linetype
中解析它们,该例程返回单个loginfo对象。在将结果写到更好的日志文件之前,我想创建这些对象的列表以进行进一步处理:
from copy import deepcopy
output_log = list()
version = -1
global_time = 0
with open(path+in_filename,'r',errors="ignore") as f:
for i,line in enumerate(f):
log_info = parse_linetype(line,version,global_time)
output_log.append(deepcopy(log_info))
# debug print:
print(i,": ",line,len(output_log),"\n",str(output_log[len(output_log)-1]),str(output_log[0]))
然而,即使我在复制列表时已经对对象进行了深度复制,此代码(对我来说也是意外的)仍会返回相同对象的列表。从解析函数正确返回了对象-实际上,作为一项额外的好处-解析的日志行中不可用的信息将从先前的日志行继承下来。但是,这种结转不应影响存储在output_log
中的对象中存储的内容。
在创建真实的Deepcopy时哪里出错了,在哪里继续使用引用,为什么?
调试打印的典型输出当前看起来像这样(并看到解析的行和返回的对象确实有所不同-但总是output_log中的所有对象都是相同的。这里的“ AIN”和“ OTHER”是其中更改的部分记录-但是第一条记录(因此用于比较的第二行)当然永远不会改变):
6288 : [10:03:04.546] AIN,57668,970,Device handling terminates
6289
-1,56797,AIN
-1,AIN
6289 : 57670: Terminating experiment loop
6290
-1,OTHER
-1,OTHER
解决方法
Deepcopy不复制类,仅复制实例。它假设所有共享类资源都是设计进行共享的,并且您将info
字典设为类属性。
这是copy
module documentation对此的看法:
此模块不复制诸如模块,方法,堆栈跟踪,堆栈框架,文件,套接字,窗口,数组或任何类似类型的类型。 通过不变地返回原始对象来“复制”功能和类(浅层和深层);这与
pickle
模块处理这些内容的方式兼容。
(加粗强调)。
将字典移动到 instance 属性:
class loginfo:
def __init__(self):
self.info = {
'global_time': -1,'exptime': 0,'type': ""
# omitted other entries for brevity
}
# ...
# assign and __str__ unchanged.
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。