2021牛客暑期多校训练营10

F Train Wreck

这也是考场上自闭的题目....
考虑栈中每个不同的状态,其实在瞎想中也想到了与图论的结合,但还是缺乏相同知识点之间的联系。如果我们考虑一个树的欧拉序的话,每个点进入时有一个标号,出来时还有一个标号。并且每个点都有这两个标号,这两个标号是一一匹配的。那这和括号匹配不就是一模一样的吗?那这道题根据括号匹配我们可以构造一棵树,...等等,本题中的括号可能构造出来的不是一个树而是一个森林,因为他可能多次出现栈为空的情况,没事我们额外加一个节点当做根节点。还这样的话就是一颗完整的树了。那题目中说到的每时每刻的栈的状态对应的是什么呢?同样考虑欧拉序的过程,其实栈中的元素都是该节点往上一直跳,直到根节点。那这样就明白了,我们要给每个点都染色,且每个点到根节点的颜色状态都不同,这样的话我们考虑一个节点x,对于x的所有儿子y而言,x到根节点的颜色对他们来说都是相同的前缀,要让他们不一样的话,只能是他们自己不一样,这样的我们就直接给他们染上不同的颜色即可。考虑染色的过程,一个颜色的数量越多显然是越不优的,所以我们染色时优先染数量多的颜色。这样之后,用不用考虑同一深度却不同父亲的节点呢?不用,因为父亲节点不同的话,在染父亲节点的时候,两个父亲节点的颜色已经不同了,所以只用考虑一个父亲节点所有的儿子即可。同一深度的就不用额外考虑了。

//不等,不问,不犹豫,不回头.
#include<bits/stdc++.h>
#define _ 0
#define ls p<<1
#define db double
#define rs p<<1|1
#define P 1000000007
#define ll long long
#define INF 1000000000
#define get(x) x=read()
#define PLI pair<ll,int>
#define PII pair<int,int>
#define ull unsigned long long
#define put(x) printf("%d\n",x)
#define putl(x) printf("%lld\n",x)
#define rep(x,y,z) for(int x=y;x<=z;++x)
#define fep(x,y,z) for(int x=y;x>=z;--x)
#define go(x) for(int i=link[x],y=a[i].y;i;y=a[i=a[i].next].y)
using namespace std;
const int N=1e6+10;
int n,cnt[N],Stack[N],top,num,ans[N],vis[N];
char c[N<<1];
bool flag=true;
vector<int>son[N]; 
priority_queue<pair<int,int> >q;

inline int read()
{
    int x=0,ff=1;
    char ch=getchar();
    while(!isdigit(ch)) {if(ch=='-') ff=-1;ch=getchar();}
    while(isdigit(ch)) {x=(x<<1)+(x<<3)+(ch^48);ch=getchar();}
    return x*ff;
}

inline void dfs(int x)
{
	if(q.size()<son[x].size()) flag=false;
	if(!flag) return;
	priority_queue<pair<int,int> >d;
	for(auto y:son[x])
	{
		ans[y]=q.top().second;
		d.push({q.top().first-1,q.top().second});
		q.pop();
	}
	while(d.size()) 
	{
		if(d.top().first>=1) q.push({d.top().first,d.top().second});
		d.pop();
	}
	for(auto y:son[x]) dfs(y);
}

int main()
{
    //freopen("1.in","r",stdin);
	get(n);scanf("%s",c+1);
	rep(i,1,n) cnt[read()]++;
	rep(i,1,n<<1)
	{
		if(c[i]=='(')//进入一个新的节点. 
		{
			++num;
			son[Stack[top]].push_back(num);
			Stack[++top]=num;
		}
		else if(c[i]==')') --top;
	}
	rep(i,1,n)
	{
		if(cnt[i]&&!vis[i])
		{
			vis[i]=1;
			q.push({cnt[i],i});
		}
	}
	dfs(0);
	if(!flag) puts("NO");
	else
	{
		puts("YES");
		rep(i,1,n) printf("%d ",ans[i]);
	}
    return (0^_^0);
}
//以吾之血,铸吾最后的亡魂.

posted @ 2021-08-19 11:05  逆天峰  阅读(57)  评论(0编辑  收藏  举报
作者:逆天峰
出处:https://www.cnblogs.com/gcfer//