python quickest way to merge dictionaries based on key match

前端 未结 3 1796
余生分开走
余生分开走 2021-01-20 16:22

I have 2 lists of dictionaries. List A is 34,000 long, list B is 650,000 long. I am essentially inserting all the List B dicts into the List A dicts based on a key match.

相关标签:
3条回答
  • 2021-01-20 16:36
    from collections import defaultdict
    dictB = defaultdict(list)
    for b in listB:
        dictB[b['ID']].append(b)
    
    for a in listA:
        a['things'] = []
        for b in dictB[a['ID']]:
            a['things'].append(b)
    

    this will turn your algorithm from O(n*m) to O(m)+O(n), where n=len(listA), m=len(listB)

    basically it avoids looping through each dict in listB for each dict in listA by 'precalculating' what dicts from listB match each 'ID'

    0 讨论(0)
  • 2021-01-20 16:42

    I'd convert ListA and ListB into dictionaries instead, dictionaries with ID as the key. Then it is a simple matter to append data using python's quick dictionary lookups:

    from collections import defaultdict
    
    class thingdict(dict):
        def __init__(self, *args, **kwargs):
            things = []
            super(thingdict,self).__init__(*args, things=things, **kwargs)
    
    A = defaultdict(thingdict)
    A[1] = defaultdict(list)
    A[2] = defaultdict(list, things=[6])  # with some dummy data
    A[3] = defaultdict(list, things=[7])
    
    B = {1: 5, 2: 6, 3: 7, 4: 8, 5: 9}
    
    for k, v in B.items():
        # print k,v
        A[k]['things'].append(v)
    
    print A
    print B
    

    This returns:

    defaultdict(<class '__main__.thingdict'>, {
        1: defaultdict(<type 'list'>, {'things': [5]}),
        2: defaultdict(<type 'list'>, {'things': [6, 6]}),
        3: defaultdict(<type 'list'>, {'things': [7, 7]}),
        4: {'things': [8]},
        5: {'things': [9]}
    })
    {1: 5, 2: 6, 3: 7, 4: 8, 5: 9}
    
    0 讨论(0)
  • 2021-01-20 16:49

    Here's an approach that may help. I'll leave it to you to fill in the details.

    Your code is slow because it is a O(n^2) algorithm, comparing every A against every B.

    If you sort each of listA and listB by id first (these are O(nlogn)) operations, then you can iterate easily through the sorted versions of A and B (this will be in linear time).

    This approach is common when you have to do external merges on very large data sets. Mihai's answer is better for internal merging, where you simply index everything by id (in memory). If you have the memory to hold these additional structures, and dictionary lookup is constant time, that approach will likely be faster, not to mention simpler. :)

    By way of example let's say A had the following ids after sorting

    acfgjp
    

    and B had these ids, again after sorting

    aaaabbbbcccddeeeefffggiikknnnnppppqqqrrr
    

    The idea is, strangely enough, to keep indexes into A and B (I know that does not sound very Pythonic). At first you are looking at a in A and a in B. So you walk through B adding all the a's to your "things" array for a. Once you exhaust the a's in B, you move up one in A, to c. But the next item in B is b, which is less than c, so you have to skip the b's. Then you arrive at a c in B, so you can start adding into "things" for c. Continue in this fashion until both lists are exhausted. Just one pass. :)

    0 讨论(0)
提交回复
热议问题