Codeforces Gym 100203D Different vectors 想法题 + Hash

博客详细介绍了Codeforces Gym 100203D问题的解决思路,重点在于如何判断两个向量是否通过旋转相等。方法是计算每个向量经过特定变换后的新值,并使用Hash技巧来O(1)确定旋转后的向量是否相等。最终,作者的代码成功实现了这一方法并通过了测试。
摘要由CSDN通过智能技术生成

题目大意:

就是现在给出N个向量, 对于两个向量他们相等的条件是可以通过轮转使得存在正整数之间的双射使得两者相等

给出N个向量问不相等的向量有多少个


大致思路:

首先对于每个向量, 先求出对于每个位置向左边(或者向右边)最近的和它相同的位置到这个位置的距离作为这个位置的新值

例如 1 2 2 3向左找的话就是 4 3 1 4, 只要两个向量这么变化之后通过轮转是一样的话就一定恩那个通过双射得到, 判断两个字符串相等的话, Hash就够了, 可以O(1)推出轮转之后的hash值


代码如下:

Result  :  Accepted     Memory  :  9176 KB     Time  :  374 ms

/*
 * Author: Gatevin
 * Created Time:  2015/8/15 19:03:34
 * File Name: Sakura_Chiyo.cpp
 */
#include<iostream>
#include<sstream>
#include<fstream>
#include<vector>
#include<list>
#include<deque>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<bitset>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cctype>
#include<cmath>
#include<ctime>
#include<iomanip>
using namespace std;
const double eps(1e-8);
typedef long long lint;
typedef unsigned long long ulint;

const ulint seed = 23333uLL;

int T, N, K;
int a[10010][110];
set<ulint> H;
ulint xp[10010];

ulint hash(int p)
{
    ulint value = a[p][0];
    for(int i = 1; i < K; i++)
        value = value*seed + a[p][i];
    return value;
}

void insert(int p, ulint value)
{
    H.insert(value);
    for(int i = 0; i < K; i++)
    {
        value = (value - a[p][i]*xp[K - 1])*seed + a[p][i];
        H.insert(value);
    }
    return;
}

int b[110];

int main()
{
    xp[0] = 1uLL;
    for(int i = 1; i < 10010; i++) xp[i] = xp[i - 1]*seed;
    scanf("%d", &T);
    while(T--)
    {
        H.clear();
        scanf("%d %d", &N, &K);
        for(int i = 0; i < N; i++)
        {
            for(int j = 0; j < K; j++)
                scanf("%d", &a[i][j]);
            for(int j = 0; j < K; j++)
            {
                b[j] = 1;
                while(a[i][j] != a[i][(j + b[j]) % K]) b[j]++;
            }
            for(int j = 0; j < K; j++) a[i][j] = b[j];
        }
        int ans = 0;
        for(int i = 0; i < N; i++)
        {
            ulint val = hash(i);
            if(H.find(val) != H.end())
                continue;
            insert(i, val);
            ans++;
        }
        printf("%d\n", ans);
    }
    return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值