A. 你相信引力吗

很明显的单调栈的题,考场上没有想到平移最大值,而是想着复制一倍序列破环成链,然后发现最大值的部分特别难维护,而且耗费时间过长,只好牺牲时间复杂度加了个 \(map\) 去重。

首先把一个最大值放到最左边,这样除了最大值都不能跨过左端点走另一半的环进行匹配

然后维护单调不增栈,对于多个连续值维护当前是第几个,往后递推即可

过程中需要特判最大值防止算重

代码实现
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e7+5;
int n,a[maxn],mxpos,sta[maxn],f[maxn],tp;
long long ans;
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
int main(){
n=read();
for(int i=1;i<=n;i++){
a[i+n]=a[i]=read();
if(a[i]>a[mxpos])mxpos=i;
}
for(int i=mxpos;i<mxpos+n;i++){
while(tp&&a[sta[tp]]<a[i])ans++,tp--;
if(a[sta[tp]]>a[i])ans++;
else ans+=f[sta[tp]]+(a[i]!=a[mxpos]);
sta[++tp]=i;
f[i]=(a[i]==a[sta[tp-1]])?f[sta[tp-1]]+1:1;
}
while(tp>2){
if(a[sta[tp]]==a[sta[2]])break;
tp--;
ans++;
}
cout<<ans;
return 0;
}

B. marshland

考虑网络流:

如果把格子黑白染色,那么一个有危险值的格子流量为 \(2\),但是费用流并不能控制其恰好为 \(2\),可能会出现流量为 \(1\) 但费用算入贡献的情况

那么可以按如下方法染色:

将棋盘染成类似上述三种颜色:

那么一种合法的方案一定由蓝到红再到黄(假定红色是危险的格子)

那么把红色格子拆点放在中间,连一条费用为危险值的边,蓝色作为左边点,原点向其连边,然后连向中间入点;黄色点放右边,从出点向其连边,然后连向汇点

注意这道题要求可行流,比起传统费用流来讲并不是有流就流,而是当前路径费用为正才流


C. party?

特产的选择方案比较奇怪,可以用网络流维护

这其实是一个二分图,左部点是 \(c\) 个人,每个人相当于拥有 \(k\) 的流量,然后每个人向其对应特产连边,特产流量为一

现在要计算满足满流的最大流量

根据 \(Hall\) 定理,二分图存在完美匹配当且仅当对于任意的 \(k\),左部任意 \(k\) 个点在右边的相邻点个数大于等于 \(k\)

由于 \(c\) 很小,直接枚举所有情况即可

对于统计一条路径上特产有哪些,可以开一个 \(bitset\),并用树剖维护

为了防止跳链浪费时间,树剖时对于完整的重链记录其所达特产,只在最后一部分线段树上查询

代码实现
#include<bits/stdc++.h>
using namespace std;
const int maxn=3e5+5;
const int len=1005;
const int inf=0x3f3f3f3f;
int n,m,id[maxn],fa[maxn],x,dep[maxn],siz[maxn],son[maxn],re[maxn],tp[maxn],tot,num,pos[maxn],q,hd[maxn],cnt,c[maxn];
bitset<len>col[maxn],S[10];
int read(){
int x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch)){
x=x*10+ch-48;
ch=getchar();
}
return x*f;
}
struct Edge{
int nxt,to;
}edge[maxn];
void add(int u,int v){
edge[++cnt].nxt=hd[u];
edge[cnt].to=v;
hd[u]=cnt;
return ;
}
void dfs(int u){
siz[u]=1;
for(int i=hd[u];i;i=edge[i].nxt){
int v=edge[i].to;
dep[v]=dep[u]+1;
dfs(v);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]])son[u]=v;
}
return ;
}
void dfs1(int u,int top){
tp[u]=top;
id[u]=++tot;
re[tot]=u;
if(u!=top)col[u]=col[fa[u]];
col[u].set(c[u]);//(1<<c[u]);
// cout<<u<<" "<<col[u]<<endl;
if(son[u])dfs1(son[u],top);
for(int i=hd[u];i;i=edge[i].nxt){
int v=edge[i].to;
if(v==son[u])continue;
dfs1(v,v);
}
return ;
}
int lca(int x,int y){
while(tp[x]!=tp[y]){
if(dep[tp[x]]<dep[tp[y]])swap(x,y);
x=fa[tp[x]];
}
if(dep[x]<dep[y])swap(x,y);
return y;
}
struct Seg{
int l,r;
bitset<len>sc;
}t[maxn*4];
void update(int p){
t[p].sc|=t[p<<1].sc|t[p<<1|1].sc;
return ;
}
void build(int p,int l,int r){
t[p].l=l;
t[p].r=r;
if(l==r){
t[p].sc.set(c[re[l]]);
// t[p].sc|=(1<<c[re[l]]);
return ;
}
int mid=l+r>>1;
build(p<<1,l,mid);
build(p<<1|1,mid+1,r);
update(p);
return ;
}
bitset<len>ask(int p,int l,int r){
if(t[p].l>=l&&t[p].r<=r){
return t[p].sc;
}
int mid=t[p].l+t[p].r>>1;
bitset<len>ans;
if(l<=mid)ans=ask(p<<1,l,r);
if(r>mid)ans|=ask(p<<1|1,l,r);
return ans;
}
bitset<len>que(int x,int ed){
bitset<len>ans;
while(tp[x]!=tp[ed]){
ans|=col[x];
x=fa[tp[x]];
}
ans|=ask(1,id[ed],id[x]);
return ans;
}
int main(){
n=read();
m=read();
q=read();
for(int i=2;i<=n;i++){
fa[i]=read();
add(fa[i],i);
}
for(int i=1;i<=n;i++)c[i]=read();
dfs(1);
dfs1(1,1);
// for(int i=1;i<=n;i++)cout<<col[i]<<endl;
build(1,1,n);
for(int i=1;i<=q;i++){
num=read();
int d;
for(int j=1;j<=num;j++){
if(j==1)d=pos[j]=read();
else d=lca(d,pos[j]=read());
}
for(int j=1;j<=num;j++){
S[j]=que(pos[j],d);
// cout<<S[j]<<endl;
}
tot=inf;
for(int T=1;T<=(1<<num)-1;T++){
bitset<len>sum;
int pnum=0;
for(int j=1;j<=num;j++){
if(T&(1<<(j-1))){
pnum++;
sum|=S[j];
}
}
int x=sum.count();
tot=min(tot,x/pnum);
}
for(int j=1;j<=num;j++)S[j].reset();
printf("%d\n",tot*num);
}
return 0;
}

D. 半夜

首先题意转化为将 \(A\) 串复制一倍,没一个子串与 \(B\) 串的 \(LCS\)

暴力 \(n^3\)

复习一下 \(LCS\) 转移方程式为 \(f[i][j]=min(f[i-1][j],f[i][j-1],f[i-1][j-1]+1)\)

考虑优化:

设 \(f_{i,j,k}=LCS(S[i,j],T[1,k])\)

很明显有 \(f_{i-1,j,k}>f_{i-1,j-1,k}\) 可以推出 \(f_{i,j,k}>f_{i,j-1,k}\)

同理 \(f_{i,j,k}>f_{i,j,k-1}\) 推出 \(f_{i-1,j,k}>f_{i-1,j,k-1}\)

那么发现 \(dp\) 的转移有分解点 \(p_{j,k}\) 和 \(q_{j,k}\) 表示:

\[f_{i,j,k}=f_{i,j−1,k}+[i≥p_{k,j}]=f_{i,j,k−1}+[i<q_{k,j}]
\]

那么推出下图(盗图 \(from\) 这篇博客):

这是 \(P<Q\) 且 \(S[j]!=T[k]\) 的情况,设 \(f_{i-1,j-1,k-1}=F\) 根据递推式推出前两行,取 \(max\) 推出第三行,发现 \(p_{j,k}=P\),\(q_{j,k}=Q\)

所有 \(S[j]=T[k]\) 的情况和这个相同

这是 \(P>Q\) 的情况,同理推出 \(p_{j,k}=Q\),\(q_{j,k}=P\)

这样两个分界点数组可以递推出来,然后实现 \(n^2\) 转移

代码实现
#include<bits/stdc++.h>
using namespace std;
const int maxn=4005;
char a[maxn],b[maxn];
int n,p[maxn][maxn],q[maxn][maxn],sum,ans;
int main(){
cin>>n;
scanf("%s",a+1);
scanf("%s",b+1);
for(int i=1;i<=n;i++)a[i+n]=a[i];
for(int i=1;i<=n*2;i++)p[i][0]=i+1,q[i][0]=1;
for(int i=1;i<=n;i++){
for(int j=1;j<=n*2;j++){
int P=p[j][i-1],Q=q[j-1][i];
if(a[j]!=b[i]&&P>Q){
p[j][i]=P,q[j][i]=Q;
}
else p[j][i]=Q,q[j][i]=P;
}
}
for(int i=1;i<=n;i++){
sum=0;
for(int j=i;j<n+i;j++){
if(i>=p[j][n])sum++;
}
ans=max(ans,sum);
}
cout<<ans;
return 0;
}

noip模拟41的更多相关文章

  1. [考试总结]noip模拟41

    发现长时间鸽博客会导致 rp--,所以今天来补一补 这个题目其实不是很毒瘤,然而是非常毒瘤... 题目不说请就是非常非常的烦人 首先 \(T1\) 就整整有两个歧义的地方,也就是说我们一共有 \(4\ ...

  2. Noip模拟41 2021.8.16

    T1 你相信引力吗 对于区间的大小关系问题,往往使用单调栈来解决 这道题的优弧和劣弧很烦,考虑将其等价的转化 由于所有的合法情况绕过的弧都不会经过最高的冰锥, 又因为环可以任意亲定起点,这样可以直接把 ...

  3. 2021.8.16考试总结[NOIP模拟41]

    T1 你相信引力吗 肯定是单调栈维护.但存在重复值,还是个环,不好搞. 发现取区间时不会越过最大值,因此以最大值为断点将环断为序列.在栈里维护当前栈中有多少个与当前元素相等的元素,小分类讨论一下. 最 ...

  4. 【HHHOJ】NOIP模拟赛 玖 解题报告

    点此进入比赛 得分: \(100+20+100=220\)(还不错) 排名: \(Rank\ 16\) \(Rating\):\(+20\) \(T1\):[HHHOJ263]「NOIP模拟赛 玖」三 ...

  5. NOIP模拟17.9.21

    NOIP模拟17.9.21 3 58 145 201 161.5 样例输出21.6 数据规模及约定对于40% 的数据,N <= 20对于60% 的数据,N <= 1000对于100% 的数 ...

  6. NOIP模拟17.9.22

    NOIP模拟17.9.22 前进![问题描述]数轴的原点上有一只青蛙.青蛙要跳到数轴上≥

  7. NOIP模拟 17.8.18

    NOIP模拟17.8.18 A.小菜一碟的背包[题目描述]Blice和阿强巴是好朋友但萌萌哒Blice不擅长数学,所以阿强巴给了她一些奶牛做练习阿强巴有 n头奶牛,每头奶牛每天可以产一定量的奶,同时也 ...

  8. NOIP 模拟4 T2

    本题属于二和一问题 子问题相互对称 考虑对于问题一:知a求b 那么根据b数组定义式 显然能发现问题在于如何求dis(最短路) 有很多算法可供选择 dijsktra,floyed,bfs/dfs,spf ...

  9. noip模拟32[好数学啊]

    noip模拟32 solutions 真是无语子,又没上100,无奈死了 虽然我每次都觉得题很难,但是还是有好多上100的 战神都200多了,好生气啊啊啊 从题开始变难之后,我的时间分配越来越不均匀, ...

随机推荐

  1. PHP5.6.6上运行 ecshop 2.7.3常见问题处理

    ecshop在在PHP5.6.6版本以后,有了很多细微的变化.而ECSHOP官方更新又太慢,发现这些问题后也不及时升级,导致用户安装使用过程中错误百出. 整理一下我遇到的问题希望对你们能有些帮组也为了 ...

  2. 使用vue实现用户管理 添加及删除功能

    简单的管理系统-增删改查 添加及删除功能 <!DOCTYPE html> <html> <head> <meta charset="UTF-8&qu ...

  3. Github连接远程仓库详细操作

    首先 咱们需要配置ssh密钥   如何生成密钥呢    我们先来看下命令 在桌面新建一个文件夹,命名规范尽量不要使用中文,然后在文件夹内  新建测试文本: 如图   在文件夹内空白处右击进入GIt b ...

  4. 获取元素在页面中位置 getBoundingClientRect()

    DOM 原生方法getBoundingClientRect()获取元素相对视口位置 DOMRect 对象包含了一组用于描述边框的只读属性--left.top.right和bottom,单位为像素.除了 ...

  5. SpringBoot-静态资源加载-源码

    目录 静态资源映射规则 什么是webjars 呢? 第二种静态资源映射规则 参考链接 静态资源映射规则 SpringBoot中,SpringMVC的web配置都在 WebMvcAutoConfigur ...

  6. Redis分布式锁的原理和实现

    前言 我们之前聊过redis的,对基础不了解的可以移步查看一下: 几分钟搞定redis存储session共享--设计实现:https://www.cnblogs.com/xiongze520/p/10 ...

  7. Android 11(R) Power HAL AIDL简析 -- 基本接口

    Android 11(R) Power HAL AIDL将分三篇文章来介绍: Android 11(R) Power HAL AIDL简析 -- 基本接口 Android 11(R) Power HA ...

  8. EZpop分析

    首先源代码如下 <?php class Modifier { protected $var; public function append($value){ include($value); } ...

  9. SQL Server中的group by(分组)

    参考网址: https://blog.csdn.net/tswc_byy/article/details/81909052 sql 查询不必需和聚合函数一起使用 ,通常来说一般是要和聚合函数一起使用来 ...

  10. linux 的删除

    1,删除 命令行 rm -rf 文件夹名称 2,下载 wget 网址 -------------------- 查找ES进程号 ps -ef | grep elastic kill -9 3250 3 ...