题目大意:

 

给你一个关系图,判断是否合法。每个人都有师父和徒弟,可以有很多个;

若A是B的师父,B是C的师父,则A也算C的师父。

不合法: 

1) . 互为师徒;(有回路) 
 2) .你的师父是你徒弟的徒弟,或者说你的徒弟是你师父的师父。(出现回路)

思路:

判断有向图中是否存在回路或至少3元环; 
此题至少有三种做法,此处更新拓扑排序的做法:

解题方法:

一:拓扑排序:

1) . 统计每个点的入度;

2) . 将入度为0的点加入队列;

3) . 出去队首元素,将此元素所连接的点入度减一,若此后入度为0则加入队列;

4) . 判断队列循环次数,若等于n则不存在3元环,则此关系图合法;

题目链接:

点击打开链接

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdlib>
#include<queue>
#include<map>
#include<stack>
#include<vector>
#include<ctime>
using namespace std;
const int N = ;
const int M = ;
int n,m;
int tot,flag;
int in[N],head[N];
struct lp
{
int u,v,nex;
lp(){}
lp(int a,int b,int c):
u(a),v(b),nex(c){}
}cw[N];
void add(int a,int b){
cw[++tot]=lp(a,b,head[a]);
head[a]=tot;
}
void tuopu(){
queue<int>Q;
while(!Q.empty())Q.pop();
for(int i=;i<n;++i){
if(in[i]==)Q.push(i);
}
int t=;
while(!Q.empty()){
t++;
int u=Q.front();Q.pop();
for(int i=head[u];i!=-;i=cw[i].nex){
int v=cw[i].v;
in[v]--;
if(in[v]==)Q.push(v);
}
}
if(t==n)flag=;
}
int main(int argc, char const *argv[])
{
int a,b;
while(~scanf("%d%d",&n,&m)&&(n)){
memset(in,,sizeof(in));
tot=-;
memset(head,-,sizeof(head));
for(int i=;i<m;++i){
scanf("%d%d",&a,&b);
add(a,b);
in[b]++;
}
flag=;
tuopu();
if(flag)printf("YES\n");
else printf("NO\n");
}
return ;
}

二:Tarjan:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<cstdlib>
#include<queue>
#include<map>
#include<stack>
#include<vector>
#include<ctime>
using namespace std;
const int N = ;
const int M = ;
int n,tot,flag,idex;
int head[N],vis[N];
int low[N],dfn[N];
int qltNum;
int qltMap[N];
stack<int>st;
struct lp{
int to,nex;
lp(){}//构造函数
lp(int a,int b):
to(a),nex(b){}
}cw[N*N];
void add(int a,int b){
cw[++tot]=lp(b,head[a]);
head[a]=tot;
}
void dfs(int u,int fa){
dfn[u]=low[u]=++idex;
vis[u]=;
st.push(u);
int v;
for(int i=head[u];i!=-;i=cw[i].nex){
v=cw[i].to;
if(v==fa){
flag=;
break;
}
if(!vis[v]){
dfs(v,u);
if(flag)return;
low[u]=min(low[u],low[v]);
}else if(vis[v]==){
low[u]=min(low[u],dfn[v]);
}
}
if(low[u]==dfn[u]){//缩点
qltNum++;
int t=;
do{
t++;
v=st.top();st.pop();
vis[v]=;
qltMap[v]=qltNum;
if(t>=){
flag=;
return;
}
}while(v!=u);
//cout<<t<<"\n";
}
}
void tarjan(){
for(int i=;i<=n;++i){
if(!vis[i]){
dfs(i,-);
}
if(flag)return;
}
}
void init(){//初始化
while(!st.empty())st.pop();
qltNum=idex=flag=;
tot=-;
memset(head,-,sizeof(head));
memset(vis,,sizeof(vis));
memset(qltMap,,sizeof(qltMap));
}
int main(int argc, char const *argv[]){
int a,b,m;
while(~scanf("%d%d",&n,&m)&&(n)){
init();
memset(head,-,sizeof(head));
for(int i=;i<m;++i){
scanf("%d%d",&a,&b);
a++,b++;
add(a,b);
}
tarjan();
if(flag)printf("NO\n");
else printf("YES\n");
}
return ;
}

最新文章

  1. VS2013 密钥 – 所有版本
  2. Quartz2D 编程指南(二)变换、图案、阴影
  3. [TypeScript] Dictionary范例
  4. OC 入门
  5. Android系列之网络(一)----使用HttpClient发送HTTP请求(通过get方法获取数据)
  6. 3DES封装类
  7. 【CodeForces 651B】Beautiful Paintings 排序+贪心
  8. Xcode 中 Git 的配置与使用
  9. html 页面 ajax 方法显示遮罩
  10. Jquery Select 下拉框处理
  11. 疑问:关于postgres的to_number()
  12. 初学HTML5的一点理解
  13. NOIP模拟赛---1.生气的LJJ (anger)
  14. LoadRunner入门(二)
  15. Qt将窗体变为顶层窗体
  16. C#设计模式之六原型模式(Prototype)【创建型】
  17. ANDROID基础ACTIVITY篇之Activity的生命周期(一)
  18. debug的一些按钮意义
  19. 如何恢复Initial commit之前的源文件
  20. Java-IO之字符输入输出流(Reader和Writer)

热门文章

  1. 笔记:Eclipse 安装 m2eclipse 插件
  2. WinSock 异步I/O模型-1
  3. WebPack介绍
  4. Redis --&gt; Ubuntu安装redis
  5. ping通但打不开网页
  6. 爬虫(scrapy中的ImagesPipeline)
  7. C语言中数据类型的取值范围
  8. DFA算法的简单说明!
  9. Django Haystack 全文检索与关键词高亮
  10. 小草手把手教你 LabVIEW 串口仪器控制——初识VISA串口