布隆过滤器用于字符串去重复,比如网络爬虫抓取时URL去重、邮件提供商反垃圾黑名单Email地址去重。等等。用哈希表也可以用于元素去重,但是占用空间比较大,而且空间使用率只有50%。
  布隆过滤器只占哈希表的1/8或1/4的空间复杂度,就能解决同样的问题,但是有一定的误判,而且不能删除已有元素。元素越多,误报率越大,但是不会漏报。对于还需要删除的布隆过滤器,还有Counter Bloom Filter,这个是布隆过滤器的变体,可以删除元素。

布隆过滤器的原理

布隆过滤器需要的是一个位数组(和位图类似)和K个映射函数(和Hash表类似),在初始状态时,对于长度为m的位数组array,它的所有位被置0。

  

对于有n个元素的集合S={S1,S2...Sn},通过k个映射函数{f1,f2,......fk},将集合S中的每个元素Sj(1<=j<=n)映射为K个值{g1,g2...gk},然后再将位数组array中相对应的array[g1],array[g2]......array[gk]置为1:

  

  如果要查找某个元素item是否在S中,则通过映射函数{f1,f2,...fk}得到k个值{g1,g2...gk},然后再判断array[g1],array[g2]...array[gk]是否都为1,若全为1,则item在S中,否则item不在S中。这个就是布隆过滤器的实现原理。

前面说到过,布隆过滤器会造成一定的误判,因为集合中的若干个元素通过映射之后得到的数值恰巧包括g1,g2,...gk,在这种情况下可能会造成误判,但是概率很小。
  总结一下,感觉布隆过滤器的实现原理并不是太复杂,但是映射函数这个东西说的还是比较虚无。对于布隆过滤器的数学证明,数学公式之类的,我还是觉得我研究不了那么深入了。毕竟现阶段水平还不到有时间去研究这么底层的,甚至于底层到和数学打交道的东西,就现阶段来说,只要知道有这样一个东西可用,会用,我就很满足了。

  本来此处留空是准备自己写个简易版的布隆过滤器的,但是实在不懂,看来数据结构还远远有待补充。

  另外附上一个第三方布隆过滤器控件,地址如下:http://bloomfilter.codeplex.com/

  DEMO:

    class Program    {        static void Main(string[] args)        {            //声明一个布隆过滤器的初始容量包含10000个项目            int capacity = 10000;            float errorRate = 0.0001F;            //初始化一个布隆过滤器            Filter
 urlSeen = new Filter
(capacity, errorRate, null);            //添加数据进入布隆过滤器            urlSeen.Add("www.baidu.com");            urlSeen.Add("www.google.com");            urlSeen.Add("www.qq.com");            Console.WriteLine("请输入网址:");            string str = Console.ReadLine();            //测试布隆过滤器是否记得这个url            if (urlSeen.Contains(str))            {                Console.WriteLine("此url已经有了!");            }            else            {                Console.WriteLine("此url尚未被收录!");            }            Console.ReadKey();        }    }