二维hash

http://csustacm.com:4803/problem/1115

一维hash是把一个字符串用一个整数表示,二维hash是把一个矩阵用一个整数表示。

for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
        hasha[i][j]=hasha[i][j-1]*base1+a[i][j];
for(int i=1;i<=n;i++)
    for(int j=1;j<=m;j++)
        hasha[i][j]+=hasha[i-1][j]*base2;

第一次是横着hash,用的是base1,此时的hash[i][j]表示的是第i行长度为j的前缀串的hash值。

第二次是竖着hash,用的是base2,此时的hash[i][j]发生了更新,此时的hash[i][j]变成了(1,1)到(i-1,j)矩阵的hash值*base2+第i行长度为j的前缀串的hash值,表示的是(1,1)到(i,j)矩阵的hash值。

然后求一个子矩阵的hash值时,比如子矩阵的右下角坐标为(i,j),行数为n,列数为m,则子矩阵的hash值为hash[i][j]-hash[i-n][j]*base1^(n)-hash[i][j-m]*base2^(m)+hash[i-n][j-m]*base1^(n)*base2^(m).(这个和二维前缀和有点类似)。

 

 

#include<bits/stdc++.h>
using namespace std;
#define ull unsigned long long
const int maxn=1005;
const ull base1=133331;
const ull base2=233;
ull hasha[maxn][maxn],hashb[maxn][maxn],pow1[maxn],pow2[maxn];
char a[maxn][maxn],b[maxn][maxn];
int n,m,t,x,y;
vector<ull>v;

int main()
{
    int ans,cas=0;
    pow1[0]=pow2[0]=1;
    for(int i=1;i<=1001;i++) pow1[i]=pow1[i-1]*base1,pow2[i]=pow2[i-1]*base2;
    while(scanf("%d%d%d%d%d",&n,&m,&t,&x,&y)!=EOF)
    {
        ans=0;
        cas++;
        v.clear();
        for(int i=1;i<=n;i++) scanf("%s",a[i]+1);
        for(int i=1;i<=n;i++)
            for(int j=1;j<=m;j++)
                hasha[i][j]=hasha[i][j-1]*base1+a[i][j];
         for(int i=1;i<=n;i++)
            for(int j=1;j<=m;j++)
                hasha[i][j]+=hasha[i-1][j]*base2;
        for(int i=x;i<=n;i++)
            for(int j=y;j<=m;j++)
                v.push_back(hasha[i][j]-hasha[i][j-y]*pow1[y]-hasha[i-x][j]*pow2[x]+hasha[i-x][j-y]*pow1[y]*pow2[x]);
        sort(v.begin(),v.end());
//       cout<<"a jz li quan bu da xiao wei x*y de jz hash val"<<endl;
//       for(int i=0;i<v.size();i++)
//           cout<<v[i]<<"  ";
//       cout<<endl;
        while(t--)
        {
            for(int i=1;i<=x;i++) scanf("%s",b[i]+1);
            for(int i=1;i<=x;i++)
                for(int j=1;j<=y;j++)
                    hashb[i][j]=hashb[i][j-1]*base1+b[i][j];
            for(int i=1;i<=x;i++)
                for(int j=1;j<=y;j++)
                    hashb[i][j]+=hashb[i-1][j]*base2;
//           cout<<"zjz hash cal "<<hashb[x][y]<<endl;
            int pos=lower_bound(v.begin(),v.end(),hashb[x][y])-v.begin();
            if(pos==v.size())
                continue;
            if(v[pos]==hashb[x][y])
                ans++;
        }
        printf("Case %d: %d\n",cas,ans);
    }
    return 0;
}

 

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在C语言中,可以通过使用二维数组来实现哈希函数。具体实现方法如下: 1. 首先,定义一个二维数组作为哈希表,数组的每个元素表示一个哈希桶,每个哈希桶可以存放多个键值对。数组的大小可以根据需要进行调整。 ```c #define MAX_SIZE 10 typedef struct { int key; int value; } Entry; Entry hashTable[MAX_SIZE][MAX_SIZE]; ``` 2. 然后,编写哈希函数来计算键值对应的哈希桶的位置。常用的哈希函数有很多种,比如直接定址法、除留余数法、平方取中法等。根据具体情况选择合适的哈希函数。 ```c int hashFunction(int key) { // 这里使用除留余数法作为哈希函数 return key % MAX_SIZE; } ``` 3. 接下来,可以编写插入和查找函数来操作哈希表。 ```c void insert(int key, int value) { int index = hashFunction(key); // 在对应的哈希桶中查找是否已经存在相同的key,如果存在,更新value;如果不存在,将键值对插入到哈希桶中 int i; for (i = 0; i < MAX_SIZE; i++) { if (hashTable[index][i].key == key) { hashTable[index][i].value = value; return; } if (hashTable[index][i].key == 0) { hashTable[index][i].key = key; hashTable[index][i].value = value; return; } } // 如果哈希桶已满,无法插入键值对 printf("Hash table is full.\n"); } int search(int key) { int index = hashFunction(key); // 在对应的哈希桶中查找key,并返回对应的value int i; for (i = 0; i < MAX_SIZE; i++) { if (hashTable[index][i].key == key) { return hashTable[index][i].value; } } // 如果未找到对应的key,返回一个特定的值表示未找到 return -1; } ``` 通过以上方法,可以实现一个简单的哈希函数和哈希表的功能。在使用哈希函数时,需要根据实际情况选择合适的哈希函数和哈希表大小,以确保哈希函数的性能和哈希表的存储能力。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C语言-一维数组和二维数组,包括动态一维数组和二维数组声明和使用](https://blog.csdn.net/u012582648/article/details/107912735)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [哈希(Hash)查找算法详解之C语言版](https://blog.csdn.net/sunnyoldman001/article/details/127345993)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值