Python将列表添加到字典的值TypeError:unhashable type:' list'

时间:2015-02-02 15:55:11

标签: python list dictionary

我正在尝试比较两个文件,然后写入字典“dns_dic”最终结果,我收到此错误:

这是我正在使用的脚本:

g  = open('/data/data/A.1/ap2014-2dom.txt','r')
f = open('/data/data/A.1/test','r')

dns_dic=defaultdict(set)
d = defaultdict(set)
psl = PublicSuffixList()

for line in g:
    line = line.strip('\n')
    domain,bl_date= line.split('|')
    bl_date = int(bl_date)
    if domain in d:
        d[domain].add(bl_date)
    else:
        d[domain] = set([bl_date])

for n, line in enumerate(f):

    line = line.strip('')
    try:
        jdata = json.loads(line)
        dom = psl.get_public_suffix(jdata.get('rrname'))
    except:
        pass
    if dom in d:
        if dom not in dns_dic:
            for i in d[domain]:
                if jdata.get('time_first') <= i <= jdata.get('time_last'):
                    ip = jdata.get('rdata') # ip one or more ips in a list 
                    if dom in dns_dic:
                        dns_dic[dom].add(ip)
                    else:
                        dns_dic[dom] = set(ip)

print dns_dic

这就是g的样子:

0001211.com|1407101455
000a.biz|1399553282
000a.biz|1400084462
000a.biz|1400243222

这就是f的样子:

{"rrname":"c.000a.biz.","time_last":1400243400,"time_first":1388645949,"rdata":["50.21.180.100"]}
{"rrname":"c.000a.biz.","time_last":1389133600,"time_first":1389133600,"rdata":["50.21.180.100"]}
{"rrname": "0001211.com.","time_last":1407101755,"time_first":1389074193,"rdata":["50.21.180.100"]}

有人知道我为什么会收到这个错误吗?我认为这是因为我正在向dns_dic添加ip(rdata)列表,但我不确定,我找不到解决方案。

1 个答案:

答案 0 :(得分:3)

您正在尝试将列表放入您的集合中:

dns_dic[dom].add(ip)

ip是此处的列表;它取自jdata

ip = jdata.get('rdata')

将其定义为列表:

"rdata":["50.21.180.100"]

集合的内容必须是可清除的,而列表则不是。

您可能希望使用set.update()来添加该列表的所有元素

dns_dic[dom].update(ip)

将变量重命名为ips(复数)会更好地反映它是一个列表。

您已在defaultdict(set)使用dns_dic个对象,因此无需测试dom in dns_dic;您可以始终使用update()

if jdata.get('time_first') <= i <= jdata.get('time_last'):
    dns_dic[dom].update(jdata.get('rdata', []))

如果您想要实际构建此类集,则需要删除以下行:

if dom not in dns_dic:

因为这意味着您忽略该域中的其他条目。

这使你的第二个循环:

for n, line in enumerate(f):
    line = line.strip('')
    try:
        jdata = json.loads(line)
    except ValueError:
        continue  # ignore invalid or empty lines
    dom = psl.get_public_suffix(jdata.get('rrname'))
    for i in d.get(domain, []):
        if jdata['time_first'] <= i <= jdata['time_last']:
            dns_dic[dom].update(jdata['rdata'])