您的位置:首页 > 其它

CodeForces - 883G(dfs)

2017-10-24 16:55 363 查看
题目:http://codeforces.com/contest/883/problem/G

题意:n点,m边,一个点s,输入m条边,有单向边和双向边,决定双向边的方向使得和s相连的点最多/最少

思路:很容易想到dfs,但是写起来很复杂,dfs的时候判断是不是双向边,是的话(最大值就是u->v,继续向下遍历;最大值就是v->u,直接截止)

代码(参考ZZY):

#include <bits/stdc++.h>
using namespace std;

const int N = 400005;
int op
,u
,v
,ans
,vis
;
vector< pair<int,int> > G
;
int sum;
void dfs1(int u)
{
if(vis[u]) return;
vis[u] = 1;
sum++;
for(auto y : G[u])
{
if(op[y.second] == 1)
dfs1(y.first);
else if(!ans[y.second])//未访问过的双向边
{
ans[y.second] = v[y.second] == y.first ? 1 : 2;//u->v 1
dfs1(y.first);
}
}
}
void dfs2(int u)
{
if(vis[u]) return;
vis[u] = 1;
sum++;
for(auto y : G[u])
{
if(op[y.second] == 1)
dfs2(y.first);
else if(!ans[y.second])//未访问过的双向边
{
ans[y.second] = v[y.second] == y.first ? 2 : 1;
}
}
}
int main()
{
int n,m,s;
cin >> n >> m >> s;
for(int i = 1;i <= m;i++)
{
scanf("%d%d%d",&op[i],&u[i],&v[i]);
G[u[i]].push_back(make_pair(v[i],i));
if(op[i] == 2)
G[v[i]].push_back(make_pair(u[i],i));
}
memset(vis,0,sizeof(vis));
memset(ans,0,sizeof(ans));
sum = 0;
dfs1(s);
printf("%d\n",sum);
for(int i = 1;i <= m;i++)
if(op[i] == 2) printf("%c",ans[i] == 1 ? '+' : '-');
printf("\n");

memset(vis,0,sizeof(vis));
memset(ans,0,sizeof(ans));
sum = 0;
dfs2(s);
printf("%d\n",sum);
for(int i = 1;i <= m;i++)
if(op[i] == 2) printf("%c",ans[i] == 1 ? '+' : '-');
printf("\n");
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: