我有两个列表,我需要将他们的元素与Borda排名进行排名。所以我做了这个功能,但我有这个错误:
TypeError: unhashable type: 'list'.
正如其他答案所示问题是我不能使用列表作为dict中的键,因为dict键需要是不可变的。所以我使用了一个元组,但错误仍然存在。
The files of the two lists look like this
list1 = [([('diritti', 'S'), ('umani', 'A')]), ([('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S'), ('umani', 'A')]), ([('forze', 'S'), ('di', 'E'), ('sicurezza', 'S')]), ([('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S')]), ([('Nazioni', 'SP'), ('Unite', 'SP')]), ([('anni', 'S'), ('di', 'E'), ('carcere', 'S')])]
list2 = [([('anni', 'S'), ('di', 'E'), ('carcere', 'S')]), ([('diritti', 'S'), ('umani', 'A')]), ([('forze', 'S'), ('di', 'E'), ('sicurezza', 'S')]), ([('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S'), ('umani', 'A')]), ([('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S')]), ([('Nazioni', 'SP'), ('Unite', 'SP')]), ([('uso', 'S'), ('eccessivo', 'A'), ('della', 'E'), ('forza', 'S')])]
我以这种方式打开这两个文件作为列表
list1 = codecs.open('/home/list1', 'r', 'utf-8').read()
list2 = codecs.open('/home/list2', 'r', 'utf-8').read()
li = ast.literal_eval(list1)
lii = ast.literal_eval(list2)
def borda_sort(lists):
###Borda’s positional ranking combines ranked lists using information of the ordinal ranks of the elements in each list.Given lists t1, t2, t3 ... tk, for each candidate c and list ti, the score B ti (c) is the number of candidates ranked below c in ti. So The total Borda score is B(c) = ∑ B ti (c) The candidates are then sorted by descending Borda scores. Given the lists = [ ['a', 'c'], ['b', 'd', 'a'], ['b', 'a', 'c', 'd'] ], the output will be ['b', 'a', 'c', 'd']
scores = {}
for l in lists:
for idx, elem in enumerate(reversed(l)):
if not elem in scores:
scores[elem] = 0
scores[elem] += idx
return sorted(scores.keys(), key=lambda elem: scores[elem], reverse=True)
lists = zip(li, lii)
print borda_sort(lists)
有人可以帮忙吗?
答案 0 :(得分:2)
使用元组而不是列表。它们可以用作词典中的键。假设列表具有固定数量的元素。即len(list)对于所有列表和常量都是相同的。
scores = {}
for l in lists:
for idx, elem in enumerate(reversed(l)):
if not elem in scores:
scores[tuple(elem)] = 0 #Notice that i have converted list to a tuple.
scores[tuple(elem)] += idx #Notice that i have converted list to a tuple.
return sorted(scores.keys(), key=lambda elem: scores[tuple(elem)], reverse=True) #Notice that i have converted list to a tuple.
答案 1 :(得分:1)
由于列表中包含复杂元素(字符串元组列表的元组),因此必须将它们转换为包含字符串元组的元组,以便它们可以清除。像这样:
list1a = [ tuple(x) for x in list1 ]
list2a = [ tuple(x) for x in list2 ]
print borda_sort([list1a, list2a])
# prints [(('diritti', 'S'), ('umani', 'A')), (('forze', 'S'), ('di', 'E'), ('sicurezza', 'S')), (('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S'), ('umani', 'A')), (('anni', 'S'), ('di', 'E'), ('carcere', 'S')), (('violazioni', 'S'), ('dei', 'E'), ('diritti', 'S')), (('Nazioni', 'SP'), ('Unite', 'SP')), (('uso', 'S'), ('eccessivo', 'A'), ('della', 'E'), ('forza', 'S'))]
OLD ANSWER 如果您有list1和list2,则调用此函数的方式为borda_sort([list1, list2])
(或borda_sort((list1, list2))
,其工作方式相同)。如果您的列表包含您想要用于borda的相同顺序的单个元素,那么答案是列出列表,而不是将列表压缩在一起。我认为zip()
没有做你认为它做的事情。 zip从一对列表中创建一对列表,如下所示:
>>> zip((1, 2, 3), ('a', 'b', 'c'))
[(1, 'a'), (2, 'b'), (3, 'c')]