2019/11/1 CSP模拟
写在前面的反思
该拿的部分分还是得拿完啊,因为懒+动作慢没有写最后一道题的菊花图和链的情况,其实这两个点并不难。。
虽然只有\(10pts\),但是已经足够往上爬一截了啊,额外的\(10pts\)在今天大众分\(210pts\)的背景下显得好重要
另外\(T2\)下来发现最后判断的地方假了,所幸好像它能跑得动的数据范围内都没出问题,但要卡还是很好卡,只是这次运气好没被卡而已,下次写的时候还是要注意,正式比赛不一定就不卡了
T1
\(sb\)题
对于\(b==1\),判断所有总数的\(gcd\)能不能整除\(a\)
对于\(b == 2\),判断总数中最小的一个开方是不是严格大于\(a\)
\(sqrt()\)的返回值是浮点数救我一命,差点想乘起来了,发现会爆\(long\ long\)
T2
思路
桶套桶
记录标本串每个字母出现的次数(第一层),设每个字母出现的次数为\(cnt_i\),那么第二层以\(cnt_i\)为下标,统计每个字母出现次数的次数,记为\(cnt1_i\)
举个例子
原标本串
\(aabbbcccd\)
第一层桶,下标对应字母:\(2\ 3\ 3\ 1\)
第二层桶,下标对应第一层桶中的值:\(1\ 1\ 2\)
考虑记录一个\(tot\)总和,当我们当前枚举到的子串和标本串的第二层桶的各项差的绝对值之和为\(0\)时,这两者第二层桶中的每个值一一对应,即可以通过变换得到
因为每次挪动区间类似滑动窗口,所以我们只需要处理最开始的区间,之后的区间减去左边挪掉的端点,加入右端新进入的端点即可继续统计答案
代码:
#include<bits/stdc++.h>
#define N (2000 + 10)
using namespace std;
inline int read() {
int cnt = 0, f = 1; char c = getchar();
while (!isdigit(c)) {if (c == '-') f = -f; c = getchar();}
while (isdigit(c)) {cnt = (cnt << 3) + (cnt << 1) + (c ^ 48); c = getchar();}
return cnt * f;
}
int T, n, Q, cnt[30], cnt1[30], g[N], f[N], q[N][30], l, r, ans, tmp, len, tot;
bool vis[N];
char ch; int s[N];
void add(int x, int sig) {
/*
因为挪动不关标本串的事,所以一切处理都在cnt1和f数组上,g只拿来比较
*/
if (f[cnt1[x]] > g[cnt1[x]]) --tot; else ++tot; //因为要减去字母x原来出现次数的影响,所以同样影响到tot,把tot被影响的部分减掉
--f[cnt1[x]]; //外层桶减去字母x原来出现次数的影响
cnt1[x] += sig; //+1或-1,看是删除还是添加,更新内层桶
if (f[cnt1[x]] >= g[cnt1[x]]) ++tot; else --tot; //更新外层桶,和上面减的地方类似
++f[cnt1[x]]; //外层桶加上字母x新出现次数的影响
}
int main() {
T = read();
while (T--) {
n = read(), Q = read();
tmp = 0;
for (; !isalpha(ch); ch = getchar());
for (; isalpha(ch); ch = getchar()) s[++tmp] = ch - 'a' + 1;
for (register int i = 1; i <= n; ++i) { //整个串,每个字母出现次数的前缀和统计
for (register int j = 1; j <= 26; ++j) q[i][j] = q[i - 1][j]; //把前一个的答案继承过来,每个字母都要继承
++q[i][s[i]]; //把这个字母的答案统计进去
} while (Q--) {
/*
cnt: 标本串的内层桶 cnt2:枚举区间的内层桶
g:标本串的外层桶 f:枚举区间的外层桶
*/
for (register int i = 1; i <= 26; ++i) cnt[i] = cnt1[i] = 0; //内层桶初始化
l = read(), r = read();
len = r - l + 1;
//当前在处理区间[1, len]这个子串(当前枚举的区间),即对于每个询问枚举的第一个区间
for (register int i = 1; i <= 26; ++i) {
cnt[i] = q[r][i] - q[l - 1][i]; //处理标本串的每个字母的出现次数,前缀和减一下
cnt1[i] = q[len][i]; //处理当前子串的每个字母的出现次数
++g[cnt[i]]; //对标本串字母出现次数的出现次数(外层桶)进行统计
++f[cnt1[i]]; //对当前子串...............进行统计(中间同上)
}
tot = 0; //总和,初始化为0
for (register int i = 1; i <= 26; ++i)
if (!vis[cnt[i]]) vis[cnt[i]] = true, tot += g[cnt[i]];
/*
这里先把所有标本串的外层桶的值累加到tot上是为了防止包含的情况,如下例子
g(标本串): 1 1 2 3
f(当前区间) : 1 1 2
这实际上是一个不合法情况,不能统计进答案,但如果我们直接枚举在子串中出现的每个字母的出现次数就会出现这样的问题,所以先在前面加好
其实和处理完之后把剩下没有被枚举到的在标本串的内层桶中出现的元素累加上tot是一个效果,但这样需要多维护一个vis数组
*/
for (register int i = 1; i <= 26; ++i) vis[cnt[i]] = false;
for (register int i = 1; i <= 26; ++i)
if (!vis[cnt1[i]]) {
vis[cnt1[i]] = true;
tot -= g[cnt1[i]]; //如果遇到在标本串和子串中同时出现的元素,先把这个减掉,防止影响后面统计答案
if (f[cnt1[i]] > g[cnt1[i]]) tot += (f[cnt1[i]] - g[cnt1[i]]);
else tot += (g[cnt1[i]] - f[cnt1[i]]);
//上面两行就是取绝对值
}
for (register int i = 1; i <= 26; ++i) vis[cnt1[i]] = false;
ans = 0;
if (!tot) ++ans; //如果总和是0,增量答案
//处理完第一个区间,开始向后挪动,下面i枚举的是右端点
for (register int i = len + 1; i <= n; ++i) {
add(s[i], 1); //加入新右端点
add(s[i - len], -1); //删除原左端点
if (!tot) ++ans; //统计答案
} printf("%d\n", ans);
for (register int i = 1; i <= 26; ++i) --g[cnt[i]], --f[cnt1[i]]; //初始化,相当于前面统计g和f的过程反过来
}
}
return 0;
}
T3
待补档,博主正在写
2019/11/1 CSP模拟的更多相关文章
- 2019/11/12 CSP模拟赛&&考前小总结
写在前面的总结 离联赛只有几天了,也马上就要回归文化课了. 有点舍不得,感觉自己的水平刚刚有点起色,却又要被抓回文化课教室了,真想在机房再赖几天啊. 像19/11/11那场的简单题,自己还是能敲出一些 ...
- 2019/11/8 CSP模拟
T1 药品实验 内网#4803 由概率定义,有\[a + b + c = 0\] 变形得到\[1 - b = a + c\] 根据题意有\[p_i = a p _{i - 1} + b p_i + c ...
- 2019/10/17 CSP模拟 总结
T1 补票 Ticket 没什么好说的,不讲了 T2 删数字 Number 很后悔的是其实考场上不仅想出了正解的方程,甚至连优化都想到了,却因为码力不足只打了\(O(n^2)\)暴力,甚至还因为细节挂 ...
- 2019.11.9 csp-s 考前模拟
2019.11.9 csp-s 考前模拟 是自闭少女lz /lb(泪奔 T1 我可能(呸,一定是唯一一个把这个题写炸了的人 题外话: 我可能是一个面向数据编程选手 作为一个唯一一个写炸T1的人,成功通 ...
- 『2019/4/9 TGDay2模拟赛 反思与总结』
2019/4/9 TGDay2模拟赛 今天是\(TG\)模拟赛的第二天了,试题难度也是相应地增加了一些,老师也说过,这就是提高组的难度了.刚开始学难的内容,一道正解也没想出来,不过基本的思路也都是对了 ...
- 『2019/4/8 TGDay1模拟赛 反思与总结』
2019/4/8 TGDay1模拟赛 这次是和高一的学长学姐们一起参加的\(TG\)模拟考,虽然说是\(Day1\),但是难度还是很大的,感觉比\(18\)年的\(Day1\)难多了. 还是看一下试题 ...
- 11.7 NOIP模拟赛
目录 2018.11.7 NOIP模拟 A 序列sequence(two pointers) B 锁lock(思路) C 正方形square(埃氏筛) 考试代码 B C 2018.11.7 NOIP模 ...
- 11/1 NOIP 模拟赛
11.1 NOIP 模拟赛 期望得分:50:实际得分:50: 思路:暴力枚举 + 快速幂 #include <algorithm> #include <cstring> #in ...
- EOJ Monthly 2019.11 E. 数学题(莫比乌斯反演+杜教筛+拉格朗日插值)
传送门 题意: 统计\(k\)元组个数\((a_1,a_2,\cdots,a_n),1\leq a_i\leq n\)使得\(gcd(a_1,a_2,\cdots,a_k,n)=1\). 定义\(f( ...
随机推荐
- Python脚本的name
- JS对象 编程练习 某班的成绩出来了,现在老师要把班级的成绩打印出来。 效果图: XXXX年XX月X日 星期X--班级总分为:81
编程练习 某班的成绩出来了,现在老师要把班级的成绩打印出来. 效果图: XXXX年XX月X日 星期X--班级总分为:81 格式要求: 1.显示打印的日期. 格式为类似"XXXX年XX月XX日 ...
- 现如今,最热门的13个Java微服务框架
曾经的 服务器领域 有许多不同的芯片架构???有哪些芯片架构???和操作系统???,经过长期发展,Java的“一次编译,到处运行”使得它在服务器领域找到一席之地,成为程序员们的最爱. 本文,我们将和大 ...
- web storem破解
话不多说 直接跳地址 http://idea.lanyus.com/ 再加个附带汉化包,使用方法已经放在压缩包里面 网盘链接: https://pan.baidu.com/s/1aEA6SSbDuRg ...
- CSIC_716_20191129【 单例模式 的五种实现方式】
单例模式 单例模式:在确定类中的属性和方法不变时,需要反复调用该类的情况. 让所有通过该类实例化出的对象,都指向同一个内存地址. 优点:节省内存空间. 单例模式有五种表现形式: 1.通过class ...
- 50. Set接口和Set的实现类HashSet
集合分类:-------------------| Collection 单列集合的根接口 ---------------| List 如果实现了List接口的集合类,具备的特点是:有序,可重复- ...
- leetcood学习笔记-160*-相交链表
题目描述: 方法一: class Solution(object): def getIntersectionNode(self, headA, headB): """ : ...
- 针对list<object>中的对象数据的一些简单处理
一 首先创建一个实体类(PersonData ): package hello; public class PersonData { String Id; String Name; String ...
- D语言简介
D语言,一种通用计算机程序语言,威力强大.功能丰富,支持多种编程范式,例如面向对象.[1] D语言最初由Digital Mars公司就职的Walter Bright于2001年发布,意图改进C++语 ...
- DLL劫持技术例子: HijackDll
控制台程序:DllLoader Dll加载器,用于动态加载目标Dll,并动态调用目标函数 #include <cstdio> #include <windows.h> type ...