迷宫城堡

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 19017    Accepted Submission(s):
8328

Problem Description
为了训练小希的方向感,Gardon建立了一座大城堡,里面有N个房间(N<=10000)和M条通道(M<=100000),每个通道都是单向的,就是说若称某通道连通了A房间和B房间,只说明可以通过这个通道由A房间到达B房间,但并不说明通过它可以由B房间到达A房间。Gardon需要请你写个程序确认一下是否任意两个房间都是相互连通的,即:对于任意的i和j,至少存在一条路径可以从房间i到房间j,也存在一条路径可以从房间j到房间i。
 
Input
输入包含多组数据,输入的第一行有两个数:N和M,接下来的M行每行有两个数a和b,表示了一条通道可以从A房间来到B房间。文件最后以两个0结束。
 
#include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
#include<stack>
using namespace std; #define MAXN 10010
#define MAXM 100010
stack<int>s;
int head[MAXN],dfn[MAXN], low[MAXN], belong[MAXM];
int instack[]; // instack[]为是否在栈中的标记数组
int n, m, cnt, scnt, top, tot;
struct Edge
{
int v, next;
}e[MAXM]; //边结点数组 void add(int u,int v)
{
e[++cnt].v=v;
e[cnt].next=head[u];
head[u]=cnt; }
void init()
{
cnt = ;
scnt=; //初始化连通分量标号
memset(head, -, sizeof(head));
memset(dfn, , sizeof(dfn)); //结点搜索的次序编号数组为0,同时可以当是否访问的数组使用
while(!s.empty()) s.pop();
} void Tarjan(int v)
{
int min,t;
dfn[v]=low[v]=cnt++;
instack[v]=;
s.push(v);
for(int i=head[v];i!=-;i=e[i].next)
{
int j=e[i].v;
if(!dfn[j])//未被访问
{
Tarjan(j);
// 更新结点v所能到达的最小次数层
if(low[v]>low[j])
low[v]=low[j];
}
else if(instack[j]&&low[v]>dfn[j])
{//如果j结点在栈内,
low[v]=dfn[j];
}
}
if(dfn[v]==low[v])
{//如果节点v是强连通分量的根
scnt++;
do
{
t=s.top();
s.pop();
instack[t]=;
belong[t]=scnt;
}
while(t!=v);
}
} int main()
{
while(scanf("%d%d",&n,&m) && (n || m))
{
init();
while(m--)
{
int u,v;
scanf("%d%d", &u, &v);
add(u,v);
}
cnt=;
for(int i = ; i <= n; i++) //枚举每个结点,搜索连通分量
{
if(!dfn[i]) //未被访问
{
Tarjan(i); //则找i结点的连通分量 }
}
if(scnt == ) printf("Yes\n"); //只有一个强连通分量,说明此图各个结点都可达
else printf("No\n");
}
return ;
}
Output
对于输入的每组数据,如果任意两个房间都是相互连接的,输出"Yes",否则输出"No"。
 
Sample Input
3 3
1 2
2 3
3 1
3 3
1 2
2 3
3 2
0 0
 
Sample Output
Yes
No
 

#include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
#include<stack>
using namespace std; #define MAXN 10010
#define MAXM 100010
stack<int>s;
int head[MAXN],dfn[MAXN], low[MAXN], belong[MAXM];
int instack[]; // instack[]为是否在栈中的标记数组
int n, m, cnt, scnt, top, tot;
struct Edge
{
int v, next;
}e[MAXM]; //边结点数组 void add(int u,int v)
{
e[++cnt].v=v;
e[cnt].next=head[u];
head[u]=cnt; }
void init()
{
cnt = ;
scnt=; //初始化连通分量标号,次序计数器,栈顶指针为0
memset(head, -, sizeof(head));
memset(dfn, , sizeof(dfn)); //结点搜索的次序编号数组为0,同时可以当是否访问的数组使用
while(!s.empty()) s.pop();
} void Tarjan(int v)
{
int min,t;
dfn[v]=low[v]=cnt++;
instack[v]=;
s.push(v);
for(int i=head[v];i!=-;i=e[i].next)
{
int j=e[i].v;
if(!dfn[j])
{
Tarjan(j);
if(low[v]>low[j])
low[v]=low[j];
}
else if(instack[j]&&low[v]>dfn[j])
{
low[v]=dfn[j];
}
}
if(dfn[v]==low[v])
{
scnt++;
do
{
t=s.top();
s.pop();
instack[t]=;
belong[t]=scnt;
}
while(t!=v);
}
} int main()
{
while(scanf("%d%d",&n,&m) && (n || m))
{
init();
while(m--)
{
int u,v;
scanf("%d%d", &u, &v);
add(u,v);
}
cnt=;
for(int i = ; i <= n; i++) //枚举每个结点,搜索连通分量
{
if(!dfn[i]) //未被访问
{
Tarjan(i); //则找i结点的连通分量 }
}
if(scnt == ) printf("Yes\n"); //只有一个强连通分量,说明此图各个结点都可达
else printf("No\n");
}
return ;
}
 #include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
using namespace std; #define MAXN 10010
#define MAXM 100010 struct Edge
{
int v, next;
}edge[MAXM]; //边结点数组 int first[MAXN], stack[MAXN], DFN[MAXN], Low[MAXN], Belong[MAXM];
// first[]头结点数组,stack[]为栈,DFN[]为深搜次序数组,Belong[]为每个结点所对应的强连通分量标号数组
// Low[u]为u结点或者u的子树结点所能追溯到的最早栈中结点的次序号
int instack[]; // instack[]为是否在栈中的标记数组
int n, m, cnt, scnt, top, tot; void init()
{
cnt = ;
scnt = top = tot = ; //初始化连通分量标号,次序计数器,栈顶指针为0
memset(first, -, sizeof(first));
memset(DFN, , sizeof(DFN)); //结点搜索的次序编号数组为0,同时可以当是否访问的数组使用
} void read_graph(int u, int v) //构建邻接表
{
edge[tot].v = v;
edge[tot].next = first[u];
first[u] = tot++;
} void Tarjan(int v) //Tarjan算法求有向图的强连通分量
{
int min, t;
DFN[v] = Low[v] = ++tot; //cnt为时间戳
instack[v] = ; //标记在栈中
stack[top++] = v; //入栈
for(int e = first[v]; e != -; e = edge[e].next)
{ //枚举v的每一条边
int j = edge[e].v; //v所邻接的边
if(!DFN[j])
{ //未被访问
Tarjan(j); //继续向下找
if(Low[v] > Low[j]) Low[v] = Low[j]; // 更新结点v所能到达的最小次数层
}
else if(instack[j] && DFN[j] < Low[v])
{ //如果j结点在栈内,
Low[v] = DFN[j];
}
}
if(DFN[v] == Low[v])
{ //如果节点v是强连通分量的根
scnt++; //连通分量标号加1
do
{
t = stack[--top]; //退栈
instack[t] = ; //标记不在栈中
Belong[t] = scnt; //出栈结点t属于cnt标号的强连通分量
}while(t != v); //直到将v从栈中退出
}
} void solve()
{
for(int i = ; i <= n; i++) //枚举每个结点,搜索连通分量
if(!DFN[i]) //未被访问
Tarjan(i); //则找i结点的连通分量
} int main()
{
while(scanf("%d%d",&n,&m) && (n || m))
{
init();
while(m--)
{
int u, v;
scanf("%d%d", &u, &v);
read_graph(u, v);
}
solve(); //求强连通分量
if(scnt == ) printf("Yes\n"); //只有一个强连通分量,说明此图各个结点都可达
else printf("No\n");
}
return ;
}
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
#include<vector> using namespace std;
vector<int>gh1[];//向量
vector<int>gh2[];
int cnt;
int vir[];
void dfs1(int n)
{
int x;
vir[n] = ;
cnt++;
for (int i = ; i<gh1[n].size(); i++)
{
x = gh1[n].at(i);
if (vir[x] == )
dfs1(x);
}
}
void dfs2(int n)
{
int x;
vir[n] = ;
cnt++;
for (int i = ; i<gh2[n].size(); i++)
{
x = gh2[n].at(i);
if (vir[x] == )
dfs2(x);
}
} int main()
{
int m, n, a, b;
while (~scanf("%d%d", &n, &m) && (m || n))
{
for (int i = ; i <= n; i++)
{
gh1[i].clear();
gh2[i].clear();
}
for (int i = ; i <= m; i++)
{
scanf("%d%d", &a, &b);
gh1[a].push_back(b);
gh2[b].push_back(a);
}
memset(vir, , sizeof(vir));
cnt = ;
dfs1();//正的走一次
if (cnt != n)
{
printf("No\n");
continue;
}
memset(vir, , sizeof(vir));
cnt = ;
dfs2();//反的走一次
if (cnt != n)
{
printf("No\n");
continue;
}
printf("Yes\n"); } return ;
}

HDU 1269 迷宫城堡(向量)(Tarjan模版题)的更多相关文章

  1. HDU 1269 迷宫城堡(强连通)

    HDU 1269 迷宫城堡 pid=1269" target="_blank" style="">题目链接 题意:中文题 思路:强连通模板题 代 ...

  2. hdu 1269 迷宫城堡

    题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=1269 迷宫城堡 Description 为了训练小希的方向感,Gardon建立了一座大城堡,里面有N个 ...

  3. HDU 1269 迷宫城堡(判断有向图强连通分量的个数,tarjan算法)

    迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  4. hdu 1269 迷宫城堡 最简单的联通图题 kosaraju缩点算法

    迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Problem Des ...

  5. HDU 1269.迷宫城堡-Tarjan or 双向DFS

    迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  6. hdu 1269 迷宫城堡 (tarjan)

    迷宫城堡Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

  7. hdu 1269 迷宫城堡(Targin算法)

    ---恢复内容开始--- 迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others ...

  8. HDU 1269 -- 迷宫城堡【有向图求SCC的数目 &amp;&amp; 模板】

    迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  9. HDU 1269 迷宫城堡 (Kosaraju)

    题目链接:HDU 1269 Problem Description 为了训练小希的方向感,Gardon建立了一座大城堡,里面有N个房间(N<=10000)和M条通道(M<=100000), ...

随机推荐

  1. 玩转X-CTR100 l STM32F4 l 基础例程printf、LED、蜂鸣器、拨码开关、位带操作

    我造轮子,你造车,创客一起造起来!塔克创新资讯[塔克社区 www.xtark.cn ][塔克博客 www.cnblogs.com/xtark/ ]      本文介绍X-CTR100控制器基础板载资源 ...

  2. MyEclipse WebSphere开发教程:WebSphere 8安装指南(二)

    [周年庆]MyEclipse个人授权 折扣低至冰点!立即开抢>> [MyEclipse最新版下载] IBM为使用WebSphere测试应用程序的开发人员提供了免费的WebSphere Ap ...

  3. Tomcat启动失败:Server Tomcat v8.0 Server at localhost was unable to start within 45 seconds

    问题如下: 解决办法: 再次重启服务器就能解决这个问题.

  4. C#读取Access数据表中某一列内容,保存至数组

    string strConn = @"Provider=Microsoft.Jet.OLEDB.4.0;Data Source=product.mdb";OleDbConnecti ...

  5. 高并发下linux ulimit优化

    系统性能一直是一个受关注的话题,如何通过最简单的设置来实现最有效的性能调优,如何在有限资源的条件下保证程序的运作,ulimit 是我们在处理这些问题时,经常使用的一种简单手段.ulimit 是一种 l ...

  6. Android Studio中设置提示函数用法

    Eclipse有一个很好的功能,就是当你代码调用某个android API时,鼠标移到对应的函数或者方法上,就会自动有一个悬浮窗提示该函数的说明(所包含的参数含义,该方法功能).迁移到Android ...

  7. 【图像处理】Haar-like特征

    特征提取的原理.代码等: 如果是白黑白,是减去一个黑的还是2个黑的,网上有不同的说法:应该需要看原论文了. 论文原文 The sum of the pixels which lie within th ...

  8. JavaScript中实现最高效的数组乱序方法

    数组乱序的意思是,把数组内的所有元素排列顺序打乱. 常用的办法是给数组原生的sort方法传入一个函数,此函数随机返回1或-1,达到随机排列数组元素的目的. 复制代码代码如下: arr.sort(fun ...

  9. .NET 之 垃圾回收机制GC

    一.GC的必要性 1.应用程序对资源操作,通常简单分为以下几个步骤:为对应的资源分配内存 → 初始化内存 → 使用资源 → 清理资源 → 释放内存. 2.应用程序对资源(内存使用)管理的方式,常见的一 ...

  10. dfs遍历痕迹的清理

    dfs 根据遍历效果分为 有痕迹遍历和无痕迹遍历, 有痕迹遍历就是对遍历过程对全局变量进行了修改的遍历(如POJ3009中的冰球问题,每次遍历会对地图造成影响),无痕迹遍历就是不对全局变量修改的遍历( ...