百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

以图搜图之“感知哈希”算法实现 感知哈希算法 matlab

liebian365 2024-10-19 07:56 22 浏览 0 评论

00、背景

Google图片搜索是Google公司于2001年7月推出的图片搜索服务。

Google总裁Eric Schmidt表示,Google图片的建立,是因为他想要看"珍妮弗·洛佩兹的绿色范思哲礼服"

2001年,Google图片编入了2.5亿张图片。2005年,这一数字增长到了十亿。到了2010年,已可搜索100亿张图片。截至2010年7月,该服务每天被访问超过一亿次。

01、感知哈希(Perceptual hashing )

感知哈希是使用指纹算法生成各种形式的多媒体的片段、哈希或指纹。

感知哈希是一种位置敏感哈希,如果多媒体的特征相似,则它是类似的。

这与加密哈希相反,加密哈希依赖于输入值的微小变化造成输出值的巨大变化的雪崩效应。

感知散列函数广泛用于查找在线版权侵权案件以及数字取证,因为散列之间具有相关性,因此可以找到相似的数据(例如具有不同的水印)。

它是基于DCT(离散余弦变换)来得到图片的hash值,具体步骤如下:

1、缩小图片:32*32是一个较好的大小,这样方便DCT计算;

2、灰度化:转换为256阶灰度图;

3、计算DCT:DCT把图片分离成分率的集合,DCT(离散余弦变换);

4、缩小DCT:DCT计算后的矩阵是32 * 32,保留左上角的8 * 8,这些代表的图片的最低频率;

5、计算平均值:计算缩小DCT后的所有像素点的平均值;

6、比较平均值:大于平均值记录为1,反之记录为0,得到phash值。

02、发展历史

Marr 和 Hildreth 1980 年的工作是该领域的开创性论文。

Christoph Zauner 2010 年 7 月的论文对该主题进行了精彩的介绍。

2016 年 6 月,Azadeh Amir Asgari 发表了有关鲁棒图像哈希欺骗的研究成果。Asgari 指出,感知哈希函数像任何其他算法一样容易出错。

研究人员在 2017 年 12 月表示,谷歌图像搜索基于感知哈希。

PHA是一类比较哈希方法的统称。图片所包含的特征被用来生成一组指纹(不过它不是唯一的),而这些指纹是可以进行比较的。

这个算法非常巧妙,无论你改变图片的高宽、亮度甚至颜色,都不会改变哈希值。

03、实现示意

04、核心代码

使用差分哈希算法计算图像的64位哈希

  /// <summary>
  /// 使用差分哈希算法计算图像的64位哈希。
  /// </summary>
  /// <param name="image">读取到的图片流</param>
  /// <returns>256位hash值</returns>
  public ulong[] DifferenceHash256(Image<Rgba32> image)
  {
      var pixels = _transformer.TransformImage(image, 17, 16);

      // 遍历像素,如果左侧像素比右侧像素亮,则将哈希设置为1。
      var hash = new ulong[4];
      var hashPos = 0;
      var hashPart = 0;
      for (var i = 0; i < 16; i++)
      {
          var rowStart = i * 17;
          for (var j = 0; j < 16; j++)
          {
              if (pixels[rowStart + j] > pixels[rowStart + j + 1])
              {
                  hash[hashPart] |= 1UL << hashPos;
              }

              if (hashPos == 63)
              {
                  hashPos = 0;
                  hashPart++;
              }
              else
              {
                  hashPos++;
              }
          }
      }

      return hash;
  }

构建索引

      private async void btnIndex_Click(object sender, EventArgs e)
      {
          if (IndexRunning)
          {
              IndexRunning = false;
              btnIndex.Text = "更新索引";
              return;
          }

          if (string.IsNullOrEmpty(txtDirectory.Text))
          {
              MessageBox.Show("请先选择文件夹");
              return;
          }

          IndexRunning = true;
          btnIndex.Text = "停止索引";
          cbRemoveInvalidIndex.Hide();
          var imageHasher = new ImageHasher(new ImageSharpTransformer());
          int? filesCount = null;
          Task.Run(() => filesCount = Directory.EnumerateFiles(txtDirectory.Text, "*", SearchOption.AllDirectories).Except(_index.Keys).Count(s => Regex.IsMatch(s, "(jpg|png|bmp)#34;, RegexOptions.IgnoreCase))).ConfigureAwait(false);
          var local = new ThreadLocal<int>(true);
          await Task.Run(() =>
          {
              var sw = Stopwatch.StartNew();
              long size = 0;
              Directory.EnumerateFiles(txtDirectory.Text, "*", SearchOption.AllDirectories).Except(_index.Keys).Where(s => Regex.IsMatch(s, "(jpg|png|bmp)#34;, RegexOptions.IgnoreCase)).Chunk(Environment.ProcessorCount * 2).AsParallel().WithDegreeOfParallelism(Environment.ProcessorCount * 2).ForAll(g =>
                {
                    foreach (var s in g)
                    {
                        if (IndexRunning)
                        {
                            if (lblProcess.InvokeRequired)
                            {
                                local.Value++;
                                lblProcess.Invoke(() => lblProcess.Text = #34;{local.Values.Sum()}/{filesCount}");
                            }
                            try
                            {
                                _index.GetOrAdd(s, _ => imageHasher.DifferenceHash256(s));
                                size += new FileInfo(s).Length;
                            }
                            catch
                            {
                                LogManager.Info(s + "格式不正确");
                            }
                        }
                        else
                        {
                            break;
                        }
                    }
                });
              lbSpeed.Text = #34;索引速度: {Math.Round(local.Values.Sum() * 1.0 / sw.Elapsed.TotalSeconds)} items/s({size * 1f / 1048576 / sw.Elapsed.TotalSeconds:N}MB/s)";
              if (cbRemoveInvalidIndex.Checked)
              {
                  foreach (var (key, _) in _index.AsParallel().WithDegreeOfParallelism(32).Where(s => !File.Exists(s.Key)))
                  {
                      _index.TryRemove(key, out _);
                  }
              }

              lbIndexCount.Text = _index.Count + "文件";
              cbRemoveInvalidIndex.Show();
              var json = JsonSerializer.Serialize(_index);
              File.WriteAllText("index.json", json, Encoding.UTF8);
              MessageBox.Show("索引创建完成,耗时:" + sw.Elapsed.TotalSeconds + "s");
          }).ConfigureAwait(false);
          IndexRunning = false;
          btnIndex.Text = "更新索引";
      }

索引结构如下:

05、最后效果

匹配度100%:

匹配度74.21%:

相关推荐

4万多吨豪华游轮遇险 竟是因为这个原因……

(观察者网讯)4.7万吨豪华游轮搁浅,竟是因为油量太低?据观察者网此前报道,挪威游轮“维京天空”号上周六(23日)在挪威近海发生引擎故障搁浅。船上载有1300多人,其中28人受伤住院。经过数天的调...

“菜鸟黑客”必用兵器之“渗透测试篇二”

"菜鸟黑客"必用兵器之"渗透测试篇二"上篇文章主要针对伙伴们对"渗透测试"应该如何学习?"渗透测试"的基本流程?本篇文章继续上次的分享,接着介绍一下黑客们常用的渗透测试工具有哪些?以及用实验环境让大家...

科幻春晚丨《震动羽翼说“Hello”》两万年星间飞行,探测器对地球的最终告白

作者|藤井太洋译者|祝力新【编者按】2021年科幻春晚的最后一篇小说,来自大家喜爱的日本科幻作家藤井太洋。小说将视角放在一颗太空探测器上,延续了他一贯的浪漫风格。...

麦子陪你做作业(二):KEGG通路数据库的正确打开姿势

作者:麦子KEGG是通路数据库中最庞大的,涵盖基因组网络信息,主要注释基因的功能和调控关系。当我们选到了合适的候选分子,单变量研究也已做完,接着研究机制的时便可使用到它。你需要了解你的分子目前已有哪些...

知存科技王绍迪:突破存储墙瓶颈,详解存算一体架构优势

智东西(公众号:zhidxcom)编辑|韦世玮智东西6月5日消息,近日,在落幕不久的GTIC2021嵌入式AI创新峰会上,知存科技CEO王绍迪博士以《存算一体AI芯片:AIoT设备的算力新选择》...

每日新闻播报(September 14)_每日新闻播报英文

AnOscarstatuestandscoveredwithplasticduringpreparationsleadinguptothe87thAcademyAward...

香港新巴城巴开放实时到站数据 供科技界研发使用

中新网3月22日电据香港《明报》报道,香港特区政府致力推动智慧城市,鼓励公私营机构开放数据,以便科技界研发使用。香港运输署21日与新巴及城巴(两巴)公司签署谅解备忘录,两巴将于2019年第3季度,开...

5款不容错过的APP: Red Bull Alert,Flipagram,WifiMapper

本周有不少非常出色的app推出,鸵鸟电台做了一个小合集。亮相本周榜单的有WifiMapper's安卓版的app,其中包含了RedBull的一款新型闹钟,还有一款可爱的怪物主题益智游戏。一起来看看我...

Qt动画效果展示_qt显示图片

今天在这篇博文中,主要实践Qt动画,做一个实例来讲解Qt动画使用,其界面如下图所示(由于没有录制为gif动画图片,所以请各位下载查看效果):该程序使用应用程序单窗口,主窗口继承于QMainWindow...

如何从0到1设计实现一门自己的脚本语言

作者:dong...

三年级语文上册 仿写句子 需要的直接下载打印吧

描写秋天的好句好段1.秋天来了,山野变成了美丽的图画。苹果露出红红的脸庞,梨树挂起金黄的灯笼,高粱举起了燃烧的火把。大雁在天空一会儿写“人”字,一会儿写“一”字。2.花园里,菊花争奇斗艳,红的似火,粉...

C++|那些一看就很简洁、优雅、经典的小代码段

目录0等概率随机洗牌:1大小写转换2字符串复制...

二年级上册语文必考句子仿写,家长打印,孩子照着练

二年级上册语文必考句子仿写,家长打印,孩子照着练。具体如下:...

一年级语文上 句子专项练习(可打印)

...

亲自上阵!C++ 大佬深度“剧透”:C++26 将如何在代码生成上对抗 Rust?

...

取消回复欢迎 发表评论: