zoukankan      html  css  js  c++  java
  • 布隆过滤器 URL去重,字符串去重


      布隆过滤器用于字符串去重复,比如网络爬虫抓取时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<string> urlSeen = new Filter<string>(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();
            }
        }

      测试如下:

      

      

  • 相关阅读:
    登录注册页面切换
    LINUX系统日常使用命令
    find命令详解
    ssh命令详解
    tar命令详解
    route命令详解
    uname命令详解
    ps命令详解
    df命令详解
    virsh命令详解
  • 原文地址:https://www.cnblogs.com/kissdodog/p/3027812.html
Copyright © 2011-2022 走看看