UCloud 的安全秘钥

题意

给出一个数组 s 串,和数组 t 串,那么如果两者长度相同且两者所含的数字全部相同,则说这两个串相似。

给定原始串 S ,以及 m 个询问 T 串,问 S 串有多少个连续子串和 T 串相似。

分析

2017年计蒜之道第五场的题目。题目很有趣,虽然比赛里只水出了中等难度。看了题解,学到了新姿势!

这道题,为了优化复杂度,就是要快速判断两个数组是否完全相同,

对于中等难度,询问的次数很少,那么我们就要加快 T 和 S 的子串的比较速度,那么可以把 1到50000 (n <= 50000)分别映射到64位无符号整型数(允许自然溢出),对于查询的串,把对应的映射后的值加起来,和 S 的子串(也是映射后的)分别比较即可。复杂度 \(O(nm)\) 。

对于困难难度,询问次数1e5,但是注意到题目给出的查询的串 T 的总长度 (len) <= 2e5 ,那么最多只有 \(\sqrt{len}\) 种长度,对于每一种长度,求出 S 的所有子串,并把这些子串和查询的串全部放到一个数组里面排序,通过遍历一遍数组即可算出答案。复杂度\(O(nlogn\sqrt{len})\) 。

我把困难难度的代码再交到中等难度题目里面,发现运行时间893ms,而 \(O(nm)\) 的算法只需要66ms,因为困难难度的代码的复杂度已经和查询的次数无关了,对于不同数据,设计不同的算法,要充分利用题目给出的信息。

code (\(O(nm)\))

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN = 5e4 + 10;
int n, m;
int a[MAXN];
int t[MAXN];
unsigned long long Z[MAXN];
int main() {
srand(timezone);
for(int i = 0; i < MAXN; i++) {
Z[i] = (unsigned long long)rand() * rand() * rand() * rand();
}
cin >> n;
for(int i = 0; i < n; i++) {
cin >> a[i];
}
cin >> m;
while(m--) {
int k, num = 0;
cin >> k;
if(k > n) {
puts("0");
continue;
}
int cnt = 0;
unsigned long long A = 1, T = 1;
for(int i = 0; i < k; i++) {
cin >> t[i];
T += Z[t[i]];
A += Z[a[i]];
}
if(A == T) {
cnt++;
}
for(int i = k; i < n; i++) {
A -= Z[a[i - k]];
A += Z[a[i]];
if(A == T) cnt++;
}
cout << cnt << endl;
}
return 0;
}

code (\(O(nlogn\sqrt{len})\))

#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ull;
const int MAXN = 1e5 + 10;
int n, m;
int s[MAXN];
ull Z[MAXN];
struct A {
int id, k;
ull num;
bool operator<(const A&other) {
return k < other.k;
}
} a[MAXN];
struct node {
int q, id;
ull num;
node() {}
node(int q_, int id_, ull num_): q(q_), id(id_), num(num_) {}
bool operator<(const node&other) {
if(num != other.num) return num < other.num;
return q > other.q;
}
} d[MAXN * 2];
int ans[MAXN * 2];
int main() {
srand(timezone);
for(int i = 0; i < MAXN; i++) {
Z[i] = (ull)rand() * rand() * rand() * rand();
}
cin >> n;
for(int i = 0; i < n; i++) {
cin >> s[i];
}
cin >> m;
for(int i = 0; i < m; i++) {
cin >> a[i].k;
a[i].id = i;
for(int j = 0; j < a[i].k; j++) {
int p;
cin >> p;
a[i].num += Z[p];
}
}
sort(a, a + m);
int l = 0, r = 0;
while(r < m) {
int cnt = 0;
d[cnt++] = node(1, a[r].id, a[r].num);
while(r < m && a[r + 1].k == a[r].k) {
r++;
d[cnt++] = node(1, a[r].id, a[r].num);
}
int k = a[r].k;
if(k > n) {
ans[a[l].id] = 0;
continue;
}
ull sum = 0;
for(int i = 0; i < k; i++) {
sum += Z[s[i]];
}
if(k) d[cnt++] = node(0, 0, sum);
for(int i = k; i < n; i++) {
sum -= Z[s[i - k]];
sum += Z[s[i]];
d[cnt++] = node(0, 0, sum);
}
sort(d, d + cnt);
int l1 = 0, r1 = 0, l2 = 0, r2 = 0;
while(l1 < cnt) {
int flg = 0;
for(int i = l1; i < cnt; i++) {
if(d[i].q) {
l1 = i;
flg = 1;
break;
}
}
if(!flg) break;
r1 = l1;
while(r1 < cnt && d[r1 + 1].q && d[r1 + 1].num == d[r1].num) r1++;
int l2 = r1 + 1, r2 = l2;
if(l2 < cnt && !d[l2].q && d[l2].num == d[r1].num) {
while(r2 < cnt && !d[r2 + 1].q && d[r2 + 1].num == d[r2].num) r2++;
for(int i = l1; i <= r1; i++) {
ans[d[i].id] = r2 - l2 + 1;
}
l1 = r2 + 1;
} else l1 = r2;
}
r++;
}
for(int i = 0; i < m; i++) {
cout << ans[i] << endl;
}
return 0;
}

(计蒜客)UCloud 的安全秘钥的更多相关文章

  1. 计蒜客 UCloud 的安全秘钥(困难)(哈希)

    UCloud 的安全秘钥(困难) 编辑代码 9.53% 1200ms 262144K 每个 UCloud 用户会构造一个由数字序列组成的秘钥,用于对服务器进行各种操作.作为一家安全可信的云计算平台,秘 ...

  2. 计蒜客 UCloud 的安全秘钥(随机化+Hash)

    题目链接 UCloud 的安全秘钥 对于简单的版本,我们直接枚举每个子序列,然后sort一下判断是否完全一样即可. #include <bits/stdc++.h> using names ...

  3. 计蒜客 UCloud 的安全秘钥 ——(hash)

    题目链接:https://nanti.jisuanke.com/t/15769. 题意是求可以变换位置以后相同的子串有多少个,那么做法是只要每个数字的平方和,立方和以及四次方和都相同就可以了. 代码如 ...

  4. 计蒜课/UCloud 的安全秘钥(hash)

    题目链接:https://nanti.jisuanke.com/t/15768 题意:中文题诶- 思路:直接hash就好了,当时zz了没想到... 代码: #include <iostream& ...

  5. 计蒜客 作弊揭发者(string的应用)

    鉴于我市拥堵的交通状况,市政交管部门经过听证决定在道路两侧安置自动停车收费系统.当车辆驶入车位,系统会通过配有的摄像头拍摄车辆画面,通过识别车牌上的数字.字母序列识别车牌,通过连接车管所车辆信息数据库 ...

  6. 计蒜客的一道题dfs

    这是我无聊时在计蒜客发现的一道题. 题意: 蒜头君有一天闲来无事和小萌一起玩游戏,游戏的内容是这样的:他们不知道从哪里找到了N根不同长度的木棍, 看谁能猜出这些木棍一共能拼出多少个不同的不等边三角形. ...

  7. 计蒜客模拟赛5 D2T1 成绩统计

    又到了一年一度的新生入学季了,清华和北大的计算机系同学都参加了同一场开学考试(因为两校兄弟情谊深厚嘛,来一场联考还是很正常的). 不幸的是,正当老师要统计大家的成绩时,世界上的所有计算机全部瘫痪了. ...

  8. 计蒜客 等边三角形 dfs

    题目: https://www.jisuanke.com/course/2291/182238 思路: 1.dfs(int a,int b,int c,int index)//a,b,c三条边的边长, ...

  9. 计蒜客 方程的解数 dfs

    题目: https://www.jisuanke.com/course/2291/182237 思路: 来自:https://blog.csdn.net/qq_29980371/article/det ...

随机推荐

  1. 基于51单片机IIC通信的AT24C02学习笔记

    引言 最近在学习几种串行通信协议,感觉收获很多,这篇文章是学习IIC总线协议的第一篇文章,以后还会再写一篇关于PCF8591 IIC通信的ADDA转换芯片的文章. 关于IIC总线 IIC 即Inter ...

  2. 解决md5不是windows平台FIPS验证的加密算法的一部分的怪异问题

    一. 发生问题 临近下班时间的下午,领导一句话:项目先上到测试服吧,我明早来看看. 我想项目还没做完,先上到测试服务器,简单看下应该是没什么问题,部署也只是一会儿的事嘛,随后把手头的项目编译,发布,拷 ...

  3. Java多线程中的单例模式

    一.在多线程环境下创建单例 方式一: package com.ietree.multithread.sync; public class Singletion { private static cla ...

  4. 关于使用mybatis的几点总结

    mybatis的几点总结 1.关于resultType和resultMap的使用 查询结果集统一采用resultType,不要采用resultMap,因为resultMap需要写许多的表字段和实体类的 ...

  5. 写具有良好风格的ABAP代码

    编程风格是一个经久不衰的话题,大家所公认的事实是:一个良好的编程风格会带来很多的好处.而对于“良好”的标准,则众说纷纭,莫衷一是.编程风格在ABAP程序中当然也有着重要的意义,因为很少看到专门针对AB ...

  6. Reactjs vs. Vuejs

    欢迎大家关注腾讯云技术社区-博客园官方主页,我们将持续在博客园为大家推荐技术精品文章哦~ 纪俊,从事Web前端开发工作,2016年加入腾讯OMG广告平台产品部,喜欢研究前端技术框架. 这里要讨论的话题 ...

  7. IOS(二)基本控件UIButton、简易动画、transform属性、UIImageView

    UIButton //1.设置UIButton 的左右移动 .center属性 获得 CGPoint 来修改x y //1.设置UIButton 的放大缩小 bounds属性 获得CGRect 然后通 ...

  8. HAProxy的三种不同类型配置方案

    haproxy是一款功能强大.灵活好用反向代理软件,提供了高可用.负载均衡.后端服务器代理的功能,它在7层负载均衡方面的功能很强大(支持 cookie track, header rewrite等等) ...

  9. Java过滤敏感词语/词汇---DFA算法

    最近网站需要在评论.投稿等地方过滤敏感词汇,于是在网上查找了相关教程,特此整理分享. 关于DFA算法,详细的可以去http://blog.csdn.net/u013378306/article/det ...

  10. 一步步学习EF Core(3.EF Core2.0路线图)

    前言 这几天一直在研究EF Core的官方文档,暂时没有发现什么比较新的和EF6.x差距比较大的东西. 不过我倒是发现了EF Core的路线图更新了,下面我们就来看看 今天我们来看看最新的EF Cor ...