P10474 [ICPC 2011 Beijing R] Matrix 矩阵哈希

题意分析

有一个01构成的大矩阵,还有一些小矩阵,问小矩阵在大矩阵中是否出现过。

求解思路

(1)暴力匹配 时间复杂度太高
(2)利用hash计算,大矩阵中每一个A*B的小矩阵的hash值,再与每一个小矩阵的hash值匹配,若有相等的,即证明找到了相同的。

原矩阵 (M×N) → 预处理所有 A×B 子矩阵的哈希值 → 存入map

查询矩阵 (A×B) → 计算查询矩阵的哈希值 → 在map中查找 → 输出结果

关键点:如何在大矩阵中计算每一个A*B的小矩阵的hash值?

这是一个二维hash的计算啊, 这时你想到了什么?是不是跟二位前缀和很像啊,借助二维前缀和的思想来解决这道题。

第一步:先计算行方向的hash值。

行方向哈希(一维前缀哈希)

cpp 复制代码
Hash[i][j] = Hash[i][j-1] × BASE1 + a[i][j]

Hash[i][j] 表示第 i 行前 j 个字符的哈希值。

第二步:再上一步的基础上,计算列方向的hash值。

这时 Hash[i][j] 表示从 (1,1) 到 (i,j) 的矩形哈希值

cpp 复制代码
Hash[i][j] = Hash[i-1][j] × BASE2 + Hash[i][j] (行哈希值)

这里为了避免冲突,行和列和BASE选择了不同的

第三步:计算A*B的子矩阵的hash值。

容斥原理

子矩阵的hash值 = Hash[i][j] - Hash[i-A][j] × Base2A - Hash[i][j-B] × Base1B + Hash[i-A][j-B] × Base2A × Base1B

BASE2a 的作用:将上面部分的权重"提升"到正确的位置

BASE1b 的作用:将左面部分的权重"提升"到正确的位置
记住: 哈希公式中的每个幂次都是为了"对齐权重",让减法和加法能够正确进行!

代码实现

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ULL;
const unsigned long long p1=131,p2=139;
const int N=1009;
unsigned long long h[N][N],h1[N][N],pow1[N],pow2[N];
int c[N][N],d[N][N];
map<ULL,bool>mapp;
int main()
{
	int a,b,n,m;
	cin>>n>>m>>a>>b;
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			scanf("%1d",&c[i][j]);
		}
	}
	//1.计算每行的hash值 
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			h[i][j]=h[i][j-1]*p1+c[i][j];
             
		}
	}
	//2.计算(1,1)到(i,j)的hash值
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			h[i][j]=h[i-1][j]*p2+h[i][j];
		}
	} 
	//3.预处理pow1,pow2的值。
	pow1[0]=pow2[0]=1; 
	for(int i=1;i<=n;i++)  pow1[i]=pow1[i-1]*p1;
	for(int i=1;i<=m;i++)  pow2[i]=pow2[i-1]*p2;
	//4.预处理出大矩阵中每一个A*B的小矩阵hash值,用map记录
	for(int i=a;i<=n;i++)
	{
		for(int j=b;j<=m;j++)
		{
			ULL res=h[i][j]-h[i-a][j]*pow2[a]-h[i][j-b]*pow1[b]+h[i-a][j-b]*pow2[a]*pow1[b];
		    mapp[res]=1;
		}
	 } 
	 //5.查询hash值是否存在
	 int q;
	 cin>>q;
	 while(q--)
	 {
	 	ULL ans=0;
	 	for(int i=1;i<=a;i++)
		{
			for(int j=1;j<=b;j++)
			{
				scanf("%1d",&d[i][j]);
			}
		}
		for(int i=1;i<=a;i++)
          for(int j=1;j<=b;j++)
            h1[i][j]=h1[i][j-1]*p1+d[i][j];
        for(int i=1;i<=a;i++)
          for(int j=1;j<=b;j++)
            h1[i][j]=h1[i-1][j]*p2+h1[i][j];
        
        ans=h1[a][b];
		if(mapp[ans])  cout<<"1"<<endl;
		else cout<<"0"<<endl;
	  } 
	 
 } 

我的bug

bug1:读入,01矩阵我用cin读的,读入错误

方法一:

cpp 复制代码
string s;
cin >> s;
c[i][j] = s[j-1] - '0';

方法二:

cpp 复制代码
scanf("%1d",&d[i][j]);

bug2:pow1[0]没有初始化

cpp 复制代码
pow1[0]=pow2[0]=1; 
for(int i=1;i<=n;i++)  pow1[i]=pow1[i-1]*p1;
for(int i=1;i<=m;i++)  pow2[i]=pow2[i-1]*p2;
相关推荐
澈2072 小时前
哈希表实战:从原理到手写实现
算法·哈希算法
旖-旎2 小时前
哈希表(存在重复元素||)(4)
数据结构·c++·算法·leetcode·哈希算法·散列表
kyle~2 小时前
工程数学---机器人变化矩阵求解
网络·矩阵·机器人
身如柳絮随风扬3 小时前
Redis中的哈希槽怎么理解
redis·哈希算法
山甫aa5 小时前
哈希集合-----从零开始的数据结构学习
数据结构·算法·哈希算法
XiYang-DING5 小时前
【Java】哈希
java·哈希算法·散列表
6Hzlia15 小时前
【Hot 100 刷题计划】 LeetCode 48. 旋转图像 | C++ 矩阵变换题解
c++·leetcode·矩阵
cccccc语言我来了19 小时前
【C++---unordered_set/map底层封装】个不拘一格的集合。它不似有序集合那般循规蹈矩,而是以一种洒脱不羁的方式,将元素们随意地散落其中。每一个元素都是独一无二的。
开发语言·c++·哈希算法
小肝一下1 天前
每日两道力扣,day8
c++·算法·leetcode·哈希算法·hot100
Tisfy1 天前
LeetCode 3740.三个相等元素之间的最小距离 I:今日先暴力,“明日“再哈希
算法·leetcode·哈希算法·题解·模拟·遍历·暴力