How to find the closest pairs (Hamming Distance) of a string of binary bins in Ruby without O^2 issues?

后端 未结 4 2082
迷失自我
迷失自我 2021-02-06 01:13

I\'ve got a MongoDB with about 1 million documents in it. These documents all have a string that represents a 256 bit bin of 1s and 0s, like:

01101010101010101101010101

相关标签:
4条回答
  • 2021-02-06 01:52

    As far as I could understand, you have an input string X and you want to query the database for a document containing string field b such that Hamming distance between X and document.b is less than some small number d.

    You can do this in linear time, just by scanning all of your N=1M documents and calculating the distance (which takes small fixed time per document). Since you only want documents with distance smaller than d, you can give up comparison after d unmatched characters; you only need to compare all 256 characters if most of them match.

    You can try to scan fewer than N documents, that is, to get better than linear time.

    Let ones(s) be the number of 1s in string s. For each document, store ones(document.b) as a new indexed field ones_count. Then you can only query documents where number of ones is close enough to ones(X), specifically, ones(X) - d <= document.ones_count <= ones(X) + d. The Mongo index should kick in here.

    If you want to find all close enough pairs in the set, see @Philippe's answer.

    0 讨论(0)
  • 2021-02-06 02:00

    The Hamming distance defines a metric space, so you could use the O(n log n) algorithm to find the closest pair of points, which is of the typical divide-and-conquer nature.

    You can then apply this repeatedly until you have "enough" pairs.

    Edit: I see now that Wikipedia doesn't actually give the algorithm, so here is one description.

    Edit 2: The algorithm can be modified to give up if there are no pairs at distance less than n. For the case of the Hamming distance: simply count the level of recursion you are in. If you haven't found something at level n in any branch, then give up (in other words, never enter n + 1). If you are using a metric where splitting on one dimension doesn't always yield a distance of 1, you need to adjust the level of recursion where you give up.

    0 讨论(0)
  • 2021-02-06 02:08

    I ended up doing a retrieval of all the documents into memory.. (subset with the id and the string).

    Then, I used a BK Tree to compare the strings.

    0 讨论(0)
  • 2021-02-06 02:11

    This sounds like an algorithmic problem of some sort. You could try comparing those with a similar number of 1 or 0 bits first, then work down through the list from there. Those that are identical will, of course, come out on top. I don't think having tons of RAM will help here.

    You could also try and work with smaller chunks. Instead of dealing with 256 bit sequences, could you treat that as 32 8-bit sequences? 16 16-bit sequences? At that point you can compute differences in a lookup table and use that as a sort of index.

    Depending on how "different" you care to match on, you could just permute changes on the source binary value and do a keyed search to find the others that match.

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