https://codeforces.com/contest/1354/problem/E
卧槽这道水题,当时kickstart开始了比较着急,被这个3迷惑住了然后没有看出来。。。不然能上一大波分
这其实就是个二分图染色,由于1,3不相连,所以互相替换是没关系的
那么对于二分图染色,首先判断有没有奇环,用dfs树向上的边判断就行了
然后这个题可能有很多连通块,对于每个联通块染色,2的数量只有两种,于是这就是个背包问题了,
就是所有连通块的2的数量加起来,能否恰好为n2,记录一下转移路径,然后最后把n3个1变成3就行了。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxl=5010;
int n,m,ans,cas,k,n1,n2,n3,tot;
int a[maxl],f[maxl],col[maxl];
int num[maxl][2],frm[maxl][maxl];
bool dp[maxl][maxl];
int dep[maxl],son[maxl],fa[maxl];
char s[maxl];
bool in[maxl],vis[maxl];
vector<int> e[maxl];
inline void dfs(int u,int st)
{
vis[u]=true;son[u]=1;f[u]=st;
num[u][0]=0;num[u][1]=1;//0->1 1->2
for(int v:e[u])
{
if(vis[v]) continue;
dep[v]=dep[u]+1;fa[v]=u;
dfs(v,st);
son[u]+=son[v];
num[u][0]+=num[v][1];
num[u][1]+=num[v][0];
}
}
inline void prework()
{
int u,v;
scanf("%d%d",&n,&m);
scanf("%d%d%d",&n1,&n2,&n3);
for(int i=1;i<=m;i++)
{
scanf("%d%d",&u,&v);
e[u].push_back(v);
e[v].push_back(u);
}
}
inline void pnt(int u,int c)
{
vis[u]=true;col[u]=c;
for(int v:e[u])
{
if(vis[v])continue;
pnt(v,3-c);
}
}
inline void mainwork()
{
dp[0][0]=true;
for(int i=1;i<=n;i++)
if(!vis[i])
{
dep[i]=1;a[++tot]=i;
dfs(i,i);
for(int j=0;j<=n2;j++)
{
if(j-num[i][0]>=0 && dp[tot-1][j-num[i][0]])
{
dp[tot][j]=true;
frm[tot][j]=0;
}
if(j-num[i][1]>=0 && dp[tot-1][j-num[i][1]])
{
dp[tot][j]=true;
frm[tot][j]=1;
}
}
}
ans=1;
for(int i=1;i<=n;i++)
for(int v:e[i])
if(dep[v]<dep[i])
{
if((dep[i]-dep[v]+1)&1)
{
ans=0;
return;
}
}
if(!dp[tot][n2])
{
ans=0;
return;
}
memset(vis,0,sizeof(vis));
int j=n2;
for(int i=tot;i>=1;i--)
if(frm[i][j]==0)
{
pnt(a[i],1);
j-=num[a[i]][0];
}
else
{
pnt(a[i],2);
j-=num[a[i]][1];
}
int num3=0;
for(int i=1;i<=n && num3<n3;i++)
if(col[i]==1)
{
num3++;
col[i]=3;
}
}
inline void print()
{
if(!ans)
puts("NO");
else
{
puts("YES");
for(int i=1;i<=n;i++)
printf("%d",col[i]);
}
}
int main()
{
int t=1;
//scanf("%d",&t);
for(cas=1;cas<=t;cas++)
{
prework();
mainwork();
print();
}
return 0;
}