在Python中查找多个列表中最相似的数字

时间:2012-09-27 23:22:50

标签: python list

在Python中,我有3个浮点数(角度)列表,范围在0-360之间,列表的长度不同。我需要找到三元组(每个列表中有一个数字),其中数字最接近。 (任何数字都不太可能是相同的,因为这是现实世界的数据。)我想用一种简单的最低标准偏差方法来衡量协议,但我不确定一个好方法实现这一点。我可以循环遍历每个列表,使用嵌套的for循环比较每个可能组合的标准偏差,并且有一个临时变量保存最好的三元组的索引,但我想知道是否有人有更好或更优雅的方式做这样的事情。谢谢!

1 个答案:

答案 0 :(得分:6)

如果有一个已建立的算法来执行此操作,我不会感到惊讶,如果是这样,您应该使用它。但我不知道一个,所以我会推测一点。

如果我必须这样做,我会尝试的第一件事就是遍历所有数字的所有可能组合,看看它需要多长时间。如果你的数据集足够小,那么发明一个聪明的算法是不值得的。为了演示设置,我将包含示例代码:

# setup
def distance(nplet):
    '''Takes a pair or triplet (an "n-plet") as a list, and returns its distance.
    A smaller return value means better agreement.'''
    # your choice of implementation here. Example:
    return variance(nplet)

# algorithm
def brute_force(*lists):
    return min(itertools.product(*lists), key = distance)

对于大型数据集,我会尝试这样的事情:首先为第一个列表中的每个数字创建一个三元组,其第一个条目设置为该数字。然后浏览这个部分填充的三元组列表,并为每个三元组从第一个列表中选择最接近该数字的第二个列表中的数字,并将其设置为三元组的第二个成员。然后浏览三元组列表,对于每个三元组,从第三个列表中选择最接近前两个数字的数字(按协议指标衡量)。最后,充分利用这一切。此示例代码演示了如何尝试使运行时保持线性列表的长度。

def item_selection(listA, listB, listC):
    # make the list of partially-filled triplets
    triplets = [[a] for a in listA]
    iT = 0
    iB = 0
    while iT < len(triplets):
        # make iB the index of a value in listB closes to triplets[iT][0]
        while iB < len(listB) and listB[iB] < triplets[iT][0]:
            iB += 1
        if iB == 0:
            triplets[iT].append(listB[0])
        elif iB == len(listB)
            triplets[iT].append(listB[-1])
        else:
            # look at the values in listB just below and just above triplets[iT][0]
            # and add the closer one as the second member of the triplet
            dist_lower = distance([triplets[iT][0], listB[iB]])
            dist_upper = distance([triplets[iT][0], listB[iB + 1]])
            if dist_lower < dist_upper:
                triplets[iT].append(listB[iB])
            elif dist_lower > dist_upper:
                triplets[iT].append(listB[iB + 1])
            else:
                # if they are equidistant, add both
                triplets[iT].append(listB[iB])
                iT += 1
                triplets[iT:iT] = [triplets[iT-1][0], listB[iB + 1]]
        iT += 1
    # then another loop while iT < len(triplets) to add in the numbers from listC
    return min(triplets, key = distance)

问题是,我可以想象这实际上找不到最好的三元组的情况,例如,如果第一个列表中的数字接近第二个列表中的一个但不接近第三个列表中的任何内容。所以你可以尝试的是为列表的所有6种可能的排序运行这个算法。我想不出一个特定的情况,那就是找不到最好的三重奏,但可能仍然存在。在任何情况下,如果你使用一个聪明的实现,假设列表已经排序,算法仍然是O(N)。

def symmetrized_item_selection(listA, listB, listC):
    best_results = []
    for ordering in itertools.permutations([listA, listB, listC]):
        best_results.extend(item_selection(*ordering))
    return min(best_results, key = distance)

另一种选择可能是计算列表1和列表2之间,列表1和列表3之间以及列表2和列表3之间所有可能的数字对。然后将所有三个对列表排序在一起,从最佳到最差协议两个数字之间。从最近的一对开始,逐个遍历列表对,只要遇到一对与您已经看过的数字共享一个数字的对,就将它们合并为三元组。对于一个合适的协议度量,一旦你找到你的第一个三元组,这将给你一个你需要迭代的最大对距离,一旦你达到它,你只需选择你最接近的三元组找到。我认为应该始终找到最好的三元组,但它将是O(N ^ 2 log N),因为需要对对的列表进行排序。

def pair_sorting(listA, listB, listC):
    # make all possible pairs of values from two lists
    # each pair has the structure ((number, origin_list),(number, origin_list))
    # so we know which lists the numbers came from
    all_pairs = []
    all_pairs += [((nA,0), (nB,1)) for (nA,nB) in itertools.product(listA,listB)]
    all_pairs += [((nA,0), (nC,2)) for (nA,nC) in itertools.product(listA,listC)]
    all_pairs += [((nB,1), (nC,2)) for (nB,nC) in itertools.product(listB,listC)]
    all_pairs.sort(key = lambda p: distance(p[0][0], p[1][0]))
    # make a dict to track which (number, origin_list)s we've already seen
    pairs_by_number_and_list = collections.defaultdict(list)
    min_distance = INFINITY
    min_triplet = None
    # start with the closest pair
    for pair in all_pairs:
        # for the first value of the current pair, see if we've seen that particular
        # (number, origin_list) combination before
        for pair2 in pairs_by_number_and_list[pair[0]]:
            # if so, that means the current pair shares its first value with
            # another pair, so put the 3 unique values together to make a triplet
            this_triplet = (pair[1][0], pair2[0][0], pair2[1][0])
            # check if the triplet agrees more than the previous best triplet
            this_distance = distance(this_triplet)
            if this_distance < min_distance:
                min_triplet = this_triplet
                min_distance = this_distance
        # do the same thing but checking the second element of the current pair
        for pair2 in pairs_by_number_and_list[pair[1]]:
            this_triplet = (pair[0][0], pair2[0][0], pair2[1][0])
            this_distance = distance(this_triplet)
            if this_distance < min_distance:
                min_triplet = this_triplet
                min_distance = this_distance
        # finally, add the current pair to the list of pairs we've seen
        pairs_by_number_and_list[pair[0]].append(pair)
        pairs_by_number_and_list[pair[1]].append(pair)
    return min_triplet

N.B。我在这个答案中编写的所有代码示例都比你在实践中做的更明确,以帮助你理解它们是如何工作的。但是当真实地做这件事时,你会使用更多的列表理解等等。

N.B.2。不保证代码可以工作:-P但它应该得到粗略的想法。