-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathKMV.hs
81 lines (64 loc) · 3.02 KB
/
KMV.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
-- Based on approach outlined in "On Synopses for Distinct-Value Estimation Under Multiset Operations"
-- (http://www.almaden.ibm.com/cs/people/peterh/cacm.pdf) and with directon from the post
-- Sketch of the Day: K-Minimum Values (http://research.neustar.biz/2012/07/09/sketch-of-the-day-k-minimum-values)
module KMV where
import qualified Data.Digest.XXHash as XXH (xxHash)
import qualified Data.PQueue.Max as DPM (MaxQueue, deleteMax, empty,
findMax, insert, null, singleton,
size, toAscList)
import Data.Word (Word32)
import System.Random
import qualified Data.Binary as DB (encode)
import qualified Data.ByteString.Lazy as BS
import qualified Data.List.Split as DLS (chunksOf)
import qualified Data.Word as DW (Word32)
mkValue :: Fractional a => BS.ByteString -> a
mkValue bs = fromIntegral h / fromIntegral (maxBound :: Word32) -- resize to {0,1}
where h = XXH.xxHash bs
mkPq :: (Fractional a, Ord a) =>
[a] -> Int -> DPM.MaxQueue a
mkPq vs k = foldr (`condInsert` k) DPM.empty vs
-- Creates priority queue using bytestring rather than already hashe values
mkPq' :: (Fractional a, Ord a) =>
[BS.ByteString] -> Int -> DPM.MaxQueue a
mkPq' bs = mkPq vs
where vs = map mkValue bs
condInsert :: (Fractional a, Ord a) => a -> Int -> DPM.MaxQueue a -> DPM.MaxQueue a
condInsert v k mq
| DPM.null mq = DPM.singleton v
| DPM.size mq < k = DPM.insert v mq -- haven't reached K so insert
| v < DPM.findMax mq = DPM.deleteMax $ DPM.insert v mq
| otherwise = mq -- value isn't larger than max so no-op
calcE :: (Fractional a, RealFrac a) => DPM.MaxQueue a -> Int -> a
calcE mq k = fromIntegral (k - 1) / DPM.findMax mq
calc :: (Fractional a, RealFrac a) => DPM.MaxQueue a -> Int -> Int
calc mq k = if sz < k then sz else round $ calcE mq k
where sz = DPM.size mq
card :: [BS.ByteString] -> Int -> Int
card bs k = calc (mkPq' bs k) k
-- Creates union of two KMV sets using the approach of taking minimum k values
-- and merging them together by preserving lowest values across both. If KMV
-- set has differing k value (length) then the resulting k value is the lesser
-- of the two.
union :: [BS.ByteString] -> [BS.ByteString] -> Int -> Int
union bs1 bs2 k = calc upq k
where pq1 = mkPq' bs1 k
pq2 = mkPq' bs2 k
uk = min (DPM.size pq1) (DPM.size pq2)
ul = take uk (DPM.toAscList pq1) ++ take uk (DPM.toAscList pq2)
upq = mkPq ul uk
main :: IO ()
main = do
let k = 1000
g <- getStdGen
let n = floor 1e6 -- Max value tested. 1e7 kills GHCI.
let c = div n 5
let r = take n (randoms g :: [DW.Word32])
let bss = DLS.chunksOf c $ map DB.encode r
let cards = map (`card` k) bss
print cards
let results = map (\x -> fromIntegral c / fromIntegral x)
print $ results cards
let unionCards = zipWith (\x y -> union x y k) bss $ tail bss
print unionCards
print $ map (* 2) $ results unionCards