noip模拟题7
T1:匹配

##思路:
首先,这道题既可以用KMP,也可以用hash
先说KMP,首先要注意的一点是:KMP的next数组求出的是boder,即既是这个串的真后缀又是真前缀,所以,对于以下类似的样例,他的输出为0,但实际是3
A:abc B:ab 增添的字符:c
所以说,要把两个串按照A在前,B在后的顺序接起来,再跳next数组。
再说hash,先求出两个串的hash数组,然后枚举长度匹配即可,注意ans初值一定要赋成0,对每一个匹配成功的长度取max即可。
T2:回家
思路
这道题我在考场上想出正解了,但是Tarjan忘了怎么打了,直接WA,这题一看就是Tarjan判割点,但要注意的是,并不是所有的割点都会是答案。考虑下面的例子,节点编号为1~10;
连边有:
(1,2),(1,3),(2,4),(3,4),(4,5),(4,10);
(5,6)(10,6),(6,7),(6,8),(7,9),(8,9);
可以发现,4,6都是割点,但答案只有4,因为6虽然是割点,但从1到10不会经过6,所以6不是必经点,由此我们发现,并不是所有割点都是必经点,要判断。
这里有一个技巧,感谢信队提供(%%信队)的技巧。
就是在Tarjan回溯的时候加一个判断,判断这个割点是否与n有关。详见代码。
上代码:
#include<bits/stdc++.h>
using namespace std;
namespace Decleration{
#define ll long long
#define rr register
const int SIZE=2e5+5;
int T,n,m;
int last[SIZE];
struct edge{int t,last;}a[SIZE<<2];
int read(){
rr int x_read=0,y_read=1;
rr char c_read=getchar();
while(c_read<'0'||c_read>'9'){
if(c_read=='-') y_read=-1;
c_read=getchar();}
while(c_read<='9'&&c_read>='0'){
x_read=(x_read<<3)+(x_read<<1)+(c_read^48);
c_read=getchar();}
return x_read*y_read;}};
using namespace Decleration;
namespace Tarjan{
int dfn[SIZE],low[SIZE];
bool pd[SIZE],dot[SIZE];
int cnt,no,root;
void tarjan(int x){
dfn[x]=low[x]=++cnt;
int flag=0;
for(int i=last[x];i!=-1;i=a[i].last){
int y=a[i].t;
if(!dfn[y]){
tarjan(y);
low[x]=min(low[x],low[y]);
if(low[y]>=dfn[x]){
flag++;
if((x!=root||flag>1)&&pd[y]) dot[x]=1;}
if(pd[y]) pd[x]=1;
//假如说x的儿子与n有关,就是说他能通到n,那他就是1,因为pd[n]是一
//pd[n]会把pd[y]更新为1,由此上推,所有能通到n的点的pd都会变成
//1,此时如果他是割点,那他就是必经点
}
else low[x]=min(low[x],dfn[y]);
}}
};
using namespace Tarjan;
void Pre_Deal(){
memset(last,-1,sizeof(last));
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(pd,0,sizeof(pd));
memset(dot,0,sizeof(dot));
cnt=no=root=0;}
int main(){
T=read();
while(T--){
n=read(),m=read();
Pre_Deal();
int num=0;
for(rr int i=1;i<=m;i++){
int u=read(),v=read();
if(u==v) continue;
a[++num].t=u;
a[num].last=last[v];
last[v]=num;
a[++num].t=v;
a[num].last=last[u];
last[u]=num;}
pd[n]=1;
tarjan(1);
for(rr int i=2;i<n;i++)
if(dot[i]) no++;
printf("%d\n",no);
if(no){
for(rr int i=2;i<n;i++)
if(dot[i]) printf("%d ",i);}
printf("\n");}
}
T3:寿司

基本思路:
首先一个贪心策略:对于每一个R我们都让他与最近的R靠近。
考虑到这是一个环,我们将转为序列,利用类似于滚动窗口的方式,将RB串循环右移,直到恢复初始的RB串,一共循环len次。
对于每一个右移出的结果,我们记l[i]为每一个R的左边的B的数量,r[i]为右边的,sumb为B的总数,sumr为R的总数。
定义x[i]=l[i]-r[i],有:
ans=min(
∑
i
=
1
s
u
m
r
\sum_{i=1}^{sumr}
∑i=1sumrmin(l[i],r[i]));
考虑求和的优化,有:
∑
i
=
1
s
u
m
r
\sum_{i=1}^{sumr}
∑i=1sumrmin(l[i],r[i])
=∑
i
=
1
s
u
m
r
\sum_{i=1}^{sumr}
∑i=1sumr((l[i]+r[i]-|x[i]|)/2)
显然有
l[i]+r[i]=sumb;
记sum=
∑
i
=
1
s
u
m
r
\sum_{i=1}^{sumr}
∑i=1sumrx[i];
原式=(sumb*sumr-sum)>>1;
那么就有:
ans=min((sumr*sumb-sum)>>1);
sumb与sumr为定值但sum要实时维护,详见代码。
上代码:
#include<bits/stdc++.h>
using namespace std;
namespace Decleration{
#define ll long long
#define rr register
const int SIZE=1e6+4;
int T;
char s[SIZE];
int x[SIZE];
int l[SIZE],r[SIZE];
int num0,num1;
inline int read(){
rr int x_read=0,y_read=1;
rr char c_read=getchar();
while(c_read<'0'||c_read>'9'){
if(c_read=='-') y_read=-1;
c_read=getchar();}
while(c_read<='9'&&c_read>='0'){
x_read=(x_read<<3)+(x_read<<1)+(c_read^48);
c_read=getchar();}
return x_read*y_read;}
};
using namespace Decleration;
int main(){
T=read();
while(T--){
memset(s,0,sizeof(s));
memset(x,0,sizeof(x));
memset(l,0,sizeof(l));
memset(r,0,sizeof(r));
scanf("%s",s+1);
int len=strlen(s+1);
int num=0;
ll sumb=0,sumr=0,sum=0;
num1=num0=0;
//num1为x[i]中负数的个数,num0为正数
priority_queue<int,vector<int>,less<int> > q;
//大根堆维护x[i]中的负值
for(rr int i=1;i<=len;i++)
if(s[i]=='B') sumb++;
else num++,l[num]=sumb,sumr++;
for(rr int i=1;i<=num;i++){
r[i]=sumb-l[i],x[i]=l[i]-r[i];
sum+=abs(x[i]);
if(x[i]>=0) num0++;//??
else num1++,q.push(x[i]);}
ll ans=LONG_LONG_MAX;
int delta=0;
//delta必须有,因为无法直接适时改变堆里的元素
//是整个q集合元素里的偏移值,就是他会加上的值
ans=min(ans,(sumb*sumr-sum)/2);
for(rr int i=len;i>=1;i--){
if(s[i]=='B'){
delta+=2;
sum+=(num0*2-num1*2);
//每个移到序列首的B都会使所有的l[i]加1,r[i]减一,则x[i]会+2
//对于负的x[i],|x[i]|-2,对于非负的x[i],|x[i]|+2;
while(!q.empty()&&q.top()+delta>=0){
num0++,num1--;
if(q.top()+delta==1) sum+=2;
//-1是特例,加2后绝对值不变,要减去
q.pop();
//大根堆的作用是适时将变为正值的x[i]排除
}
}
else num0--,num1++,q.push(-sumb-delta);
//delta会一直递增下去,所以-sumb要-delta
ans=min(ans,(sumb*sumr-sum)/2);
}
printf("%lld\n",ans);
}
}
2021.6.11 现役
noip模拟题7的更多相关文章
- 【入门OJ】2003: [Noip模拟题]寻找羔羊
这里可以复制样例: 样例输入: agnusbgnus 样例输出: 6 这里是链接:[入门OJ]2003: [Noip模拟题]寻找羔羊 这里是题解: 题目是求子串个数,且要求简单去重. 对于一个例子(a ...
- NOIP模拟题汇总(加厚版)
\(NOIP\)模拟题汇总(加厚版) T1 string 描述 有一个仅由 '0' 和 '1' 组成的字符串 \(A\),可以对其执行下列两个操作: 删除 \(A\)中的第一个字符: 若 \(A\)中 ...
- 9.9 NOIP模拟题
9.9 NOIP模拟题 T1 两个圆的面积求并 /* 计算圆的面积并 多个圆要用辛普森积分解决 这里只有两个,模拟计算就好 两圆相交时,面积并等于中间两个扇形面积减去两个三角形面积 余弦定理求角度,算 ...
- 8.22 NOIP 模拟题
8.22 NOIP 模拟题 编译命令 g++ -o * *.cpp gcc -o * *.c fpc *.pas 编译器版本 g++/gcc fpc 评测环境 位 Linux, .3GHZ CPU ...
- NOIP模拟题17.9.26
B 君的任务(task)[题目描述]与君初相识,犹如故人归.B 君看到了Z 君的第一题,觉得很难.于是自己出了一个简单题.你需要完成n 个任务,第i 任务有2 个属性ai; bi.其中ai 是完成这个 ...
- noip模拟题题解集
最近做模拟题看到一些好的题及题解. 升格思想: 核电站问题 一个核电站有N个放核物质的坑,坑排列在一条直线上.如果连续M个坑中放入核物质,则会发生爆炸,于是,在某些坑中可能不放核物质. 任务:对于给定 ...
- NOIP 模拟题
目录 T1 : grid T2 : ling T3 : threebody 数据可私信我. T1 : grid 题目:在一个\(n*n\)的方格中,你只能斜着走.为了让问题更简单,你还有一次上下左右走 ...
- 9.22 NOIP模拟题
吉林省信息学奥赛 2017 冬令营 ...
- 6.19 noip模拟题(题目及解析转自 hzwer 2014-3-15 NOIP模拟赛)
Problem 1 高级打字机(type.cpp/c/pas) [题目描述] 早苗入手了最新的高级打字机.最新款自然有着与以往不同的功能,那就是它具备撤销功能,厉害吧. 请为这种高级打字机设计一个程序 ...
- noip模拟题《迷》enc
[问题背景]zhx 和他的妹子聊天.[问题描述] 考虑一种简单的加密算法. 假定所有句子都由小写英文字母构成, 对于每一个字母, 我们将它唯一地映射到另一个字母.例如考虑映射规则:a- ...
随机推荐
- Bugku-web-md5 collision(NUPT_CTF)
总结了两道MD5绕过的题目. 根据MD5的特性,有两点漏洞 1.两个开头为0的md5值相同. 2.md5不能处理数组. 3.==用法,0 == 字符串是成立的,从而可以绕过MD5检查. 根据特性,我们 ...
- 100的累加和 while 循环
//100的累加和 while 循环 #include <stdio.h> int main() { int sum = 0; //5050 int i = 0; while(i < ...
- Kotlin强化实战!这份学习手册让你的面试稳如泰山
一.引言 正如官网的slogan所描述:kotlin,是一门让程序员写代码时更有幸福的现代语言. 同时,也正如维基百科里介绍: JetBrains公司希望Kotlin能够推动IntelliJ IDEA ...
- MySQL学习01(初识MySQL)
初识MySQL 只会写代码的是码农:学好数据库,基本能混口饭吃:在此基础上再学好操作系统和计算机网络,就能当一个不错的程序员.如果能再把离散数学.数字电路.体系结构.数据结构/算法.编译原理学通透,再 ...
- Python 应用爬虫下载酷狗音乐
应用爬虫下载酷狗音乐 首先我们需要进入到这个界面 想要爬取这些歌曲链接,然而这个是一个假的网站,虽然单机右键进行检查能看到这些歌曲的链接,可进行爬取时,却爬取不到这些信息. 这个时候我们就应该换一种思 ...
- 分享一个自己制作的XML在线编辑器
前言 一年多没更新博客了,原因是疫情期间<骑马与砍杀2>发售,然后去写游戏MOD去了. 用C#大概写了7个月的游戏MOD,每天晚上肝到很晚,然后期间又因为介绍这个游戏MOD,学习了PR,然 ...
- Mysql使用存储过程快速添加百万数据
前言 为了体现不加索引和添加索引的区别,需要使用百万级的数据,但是百万数据的表,如果使用一条条添加,特别繁琐又麻烦,这里使用存储过程快速添加数据,用时大概4个小时. 创建一个用户表 CREATE TA ...
- 【Python机器学习实战】决策树和集成学习(二)——决策树的实现
摘要:上一节对决策树的基本原理进行了梳理,本节主要根据其原理做一个逻辑的实现,然后调用sklearn的包实现决策树分类. 这里主要是对分类树的决策进行实现,算法采用ID3,即以信息增益作为划分标准进行 ...
- NOIP 模拟 10 考试总结
T1 一道很妙的题,打暴力分也很多,但是考试的时候忘开 long long 了. 题解 T2 一道挺水的题,不过...(打挂了) 题解 T3 此题甚妙,转化真多,不过对思维是一个非常大的扩展 题解 考 ...
- 简单实现 nodejs koa2 mysql 增删改查 制作接口
1.首先 在电脑上安装 nodejs (此处略过) 2.全局安装 koa2 (这里使用的淘宝镜像cnpm,有兴趣的同学可以自行搜索下) cnpm install koa-generator -g 3. ...