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;
相关推荐
爱喝纯牛奶的柠檬18 小时前
基于STM32的4*4矩阵软键盘驱动
stm32·嵌入式硬件·矩阵
Frostnova丶18 小时前
LeetCode 48 & 1886.矩阵旋转与判断
算法·leetcode·矩阵
Trouvaille ~20 小时前
【优选算法篇】哈希表——空间换时间的极致艺术
c++·算法·leetcode·青少年编程·蓝桥杯·哈希算法·散列表
楼田莉子1 天前
C++高并发内存池:内存池调优与测试
c++·后端·哈希算法·visual studio
阿Y加油吧1 天前
力扣打卡——搜索二维矩阵、相交链表
线性代数·leetcode·矩阵
小白自救计划1 天前
力扣知识点杂集
算法·leetcode·哈希算法
qq_283720051 天前
WebGL基础教程(十四):投影矩阵深度解析——正交 vs 透视,从公式推导到实战
线性代数·矩阵·webgl·正交·投影
We་ct1 天前
LeetCode 74. 搜索二维矩阵:两种高效解题思路
前端·算法·leetcode·矩阵·typescript·二分查找
cui_ruicheng1 天前
C++ 数据结构进阶:unordered_map 与 unordered_set源码分析与实现
数据结构·c++·算法·哈希算法