1

我正在尝试在 python 中构建一个函数,如果来自dict1的特定值与dict2. 我的功能如下所示:

def dict_matcher(dict1, dict2, item1_pos, item2_pos):
"""Uses a tuple value from dict1 to search for a matching tuple value in dict2. If a match is found, the other values from dict1 and dict2 are returned."""
for item1 in dict1:
    for item2 in dict2:
        if dict1[item1][item1_pos] == dict2[item2][item2_pos]:
            yield(dict1[item1][2], dict2[item2][6])

我是这样使用dict_matcher的:

matches = [myresults for myresults in dict_matcher(dict1, dict2 , 2, 6)]
print(matches)

当我打印时,matches我得到一个正确匹配 dict1 和 dict2 值的列表,如下所示:

[('frog', 'frog'), ('spider', 'spider'), ('cricket', 'cricket'), ('hampster', 'hampster')]

如何向此函数添加变量参数,以便除了打印每个字典中的匹配值之外,我还可以在dict1[item1][2] and dict2[item2][6]匹配的实例中打印每个字典项的其他值?我可以使用 *args 吗?谢谢您的帮助。

编辑:好的,我想做什么似乎有些混乱,所以让我试试另一个例子。

dict1 = {1: ('frog', 'green'), 2: ('spider', 'blue'), 3: ('cricket', 'red')}

dict2 = {a: ('frog', 12.34), b: ('ape', 22.33), c: ('lemur', 90.21)}

dict_matcher(dict1, dict2, 0, 0)将从 dict1 中找到 value[0] 和从 dict2 中找到 value[0] 的匹配值。在这种情况下,唯一的匹配是'frog'。我上面的功能就是这样做的。我想要做的是扩展函数,以便能够从dict1[value][0] == dict2[value][0]我希望在函数参数中指定的字典项中打印出其他值。

4

3 回答 3

1

您可以使用切片对象

def dict_matcher(dict1, dict2, pos1, pos2, slicer1=(), slicer2=()):
    slice1 = slice(*slicer1) if slicer1 else slice(len(dict1))
    slice2 = slice(*slicer2) if slicer2 else slice(len(dict2))
    for data1 in dict1.values():
        for data2 in dict2.values():
            if data1[pos1] == data2[pos2]:
                yield data1[slice1], data2[slice2]

for result1, result2 in dict_matcher(my_dict, your_dict, 2, 6, (3, 8, 2), (2, 6)):
    print result1, result2
  • some_list[slice(3, 8, 2)]相当于some_list[3:8:2],为您some_list提供从第四个元素(索引为 3)开始到第八个元素的每个第二个元素。
  • some_list[slice(2, 6)]相当于some_list[2:6],为您some_list提供从第三个元素(索引为 2)开始到第六个元素的每个元素。
  • some_list[slice(7)]相当于some_list[:7],为您提供some_list最多第七个元素的每个元素。

如果您省略slicer1/2参数,该函数假定您想要整个列表并相应地设置切片。

此外,我删除了不必要的字典查找。

于 2011-08-11T01:47:20.160 回答
0

你说你称它为

matches = [myresults for myresults in dict_matcher(dict1, dict2 , 2, 6)]

你应该把它称为

matches = list(dict_matcher(dict1, dict2 , 2, 6))

它的签名是

def dict_matcher(dict1, dict2, item1_pos, item2_pos, *args):

所以通过了 4 个参数和 4 个命名参数。所以*args结果args = None

我不确定你到底想要什么,但如果你这样做

yield dict1[item1][item1_pos], dict2[item2][item2_pos]

你会得到和你做的一样的东西

yield dict1[item1][2], dict2[item2][6]

如果您想获得整个匹配项,请执行

yield dict1[item1], dict2[item2]

如果您想从每个项目中获取一个项目,但不是匹配项目,请执行

def dict_matcher(dict1, dict2, item1_pos, item2_pos, other1_pos, other2_pos):

yield dict1[item1][other1_pos], dict2[item2][other2_pos]

matches = list(dict_matcher(dict1, dict2 , 2, 6, 3, 8)) 

或任何代替3和8的东西。

如果你想得到几个,但不是所有的项目,做

def dict_matcher(dict1, dict2, item1_pos, item2_pos, other1_poss, other2_poss):

yield [dict1[item1][i] for i in other1_poss], [dict2[item2][i] for i in other2_poss]

matches = list(dict_matcher(dict1, dict2 , 2, 6, (2, 3), (6, 8))) 

或任何代替 [2, 3] 和 [3, 8] 的东西。

如果这不是你的意思,请告诉我。

于 2011-08-11T01:46:47.103 回答
0

您是否希望指定任意数量的索引对来尝试比较匹配?

像这样的东西?

def matcher(d1, d2, *args):
    indexes = zip(args[0::2], args[1::2])
    for a, b in indexes:
        for value1 in dict1.values():
            for value2 in dict2.values():
                x, y = value1[a], value2[b]
                    if x == y:
                        yield x, y

dict1 = {1: ('frog', 'green'), 2: ('spider', 'blue'), 3: ('cricket', 'red')}
dict2 = {a: ('frog', 12.34), b: ('ape', 22.33), c: ('lemur', 90.21)}

matches = list(matcher(d1, d2, 
                        0, 0, # first pair to search for matches
                        1, 1  # second pair, 
                        # ... and so on,
))

print matches
# [('frog', 'frog')]

我不认为这是非常有用的,但这会做到。由于切片的魔力,如果您指定奇数个参数,这仍然可以工作。但是,正因为如此,匹配器函数的接口很容易被错误地使用。

我强烈认为更像是:

def matcher(d1, d2, indexes_to_check):
    ...
print list(matcher(d1, d2, [(0, 0), (1, 1), ...]))

def matcher(d1, d2, *indexes_to_check):
    ...
print list(matcher(d1, d2, (0, 0), (1, 1)))
于 2011-08-11T04:07:18.503 回答