P7124 Ynoi2008 stcm
P7124 Ynoi2008 stcm
妙妙构造。
思路
求出树的 dfn
序,进行分治,对于
每次处理的是
进入下一段前,删除该段在集合中的部分并将该段的补集全部加入集合。
退出这一段时清除自己在集合中的部分。
命令次数是
CODE
#include<bits/stdc++.h>
using namespace std;
const int maxn=1e5+5;
struct Edge
{
int tot;
int head[maxn];
struct edgenode{int to,nxt;}edge[maxn*2];
inline void add(int x,int y)
{
tot++;
edge[tot].to=y;
edge[tot].nxt=head[x];
head[x]=tot;
}
}T;
int n,cok;
int dfn[maxn],ed[maxn],fdfn[maxn];
vector<int>vec;
inline void clr()
{
for(int i=1;i<=n;i++) T.head[i]=dfn[i]=ed[i]=fdfn[i]=0;
T.tot=cok=0;
vec.clear();
}
inline void dfs(int u)
{
ed[u]=dfn[u]=++cok;fdfn[cok]=u;
vec.push_back(dfn[u]);
for(int i=T.head[u];i;i=T.edge[i].nxt)
{
int v=T.edge[i].to;
dfs(v);
ed[u]=ed[v];
}
}
inline void solve(int l,int r)
{
vector<int>lv,rv;
int ld=l-1,rd=r+1,mid=(l+r)>>1,cnt=0;
for(auto v:vec)
{
if(ed[fdfn[v]]<mid) lv.push_back(v);
else if(v>mid) rv.push_back(v);
else
{
while(ld<v-1) printf("+%d",fdfn[++ld]),cnt++;
while(rd>ed[fdfn[v]]+1) printf("+%d",fdfn[--rd]),cnt++;
printf("=%d",fdfn[v]);
}
}
if(l==r) return ;
for(int i=1;i<=cnt;i++) printf("-");
if(l==r) return ;
for(int i=l;i<=mid;i++) printf("+%d",fdfn[i]);
swap(vec,rv);
solve(mid+1,r);
for(int i=l;i<=mid;i++) printf("-");
for(int i=mid+1;i<=r;i++) printf("+%d",fdfn[i]);
swap(vec,lv);
solve(l,mid);
for(int i=mid+1;i<=r;i++) printf("-");
}
int main()
{
while(scanf("%d",&n)!=EOF)
{
clr();
for(int i=1;i<n;i++)
{
int u;
scanf("%d",&u);
T.add(u,i+1);
}
dfs(1);
sort(vec.begin(),vec.end());
solve(1,n);
printf("!\n");
}
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现