首页 > 其他分享 >街道赛跑

街道赛跑

时间:2023-04-04 23:00:15浏览次数:41  
标签:赛跑 终点 int 路口 跑道 街道 起点

街道赛跑 洛谷

题目描述

图一表示一次街道赛跑的跑道。可以看出有一些路口(用 \(0\) 到 \(N\) 的整数标号),和连接这些路口的箭头。路口 \(0\) 是跑道的起点,路口 \(N\) 是跑道的终点。箭头表示单行道。运动员们可以顺着街道从一个路口移动到另一个路口(只能按照箭头所指的方向)。当运动员处于路口位置时,他可以选择任意一条由这个路口引出的街道。

图一:有 \(10\) 个路口的街道

一个良好的跑道具有如下几个特点:

  1. 每一个路口都可以由起点到达。
  2. 从任意一个路口都可以到达终点。
  3. 终点不通往任何路口。

运动员不必经过所有的路口来完成比赛。有些路口却是选择任意一条路线都必须到达的(称为“不可避免”的)。在上面的例子中,这些路口是 \(0\),\(3\),\(6\),\(9\)。对于给出的良好的跑道,你的程序要确定“不可避免”的路口的集合,不包括起点和终点。

假设比赛要分两天进行。为了达到这个目的,原来的跑道必须分为两个跑道,每天使用一个跑道。第一天,起点为路口 \(0\),终点为一个“中间路口”;第二天,起点是那个中间路口,而终点为路口 \(N\)。对于给出的良好的跑道,你的程序要确定“中间路口”的集合。如果良好的跑道 \(C\) 可以被路口 \(S\) 分成两部分,这两部分都是良好的,并且 \(S\) 不同于起点也不同于终点,同时被分割的两个部分满足下列条件:(1)它们之间没有共同的街道(2)\(S\) 为它们唯一的公共点,并且 \(S\) 作为其中一个的终点和另外一个的起点。那么我们称 \(S\) 为“中间路口 ”。在例子中只有路口 \(3\) 是中间路口。


输入格式

输入文件包括一个良好的跑道,最多有 \(50\) 个路口,\(100\) 条单行道。

共有 \(N+2\) 行,前面 \(N+1\) 行中第 \(i\) 行表示以编号为 \(i-1\) 的路口作为起点的街道,每个数字表示一个终点。行末用 -2 作为结束。最后一行只有一个数字 -1


输出格式

第一行:跑道中“不可避免的路口”的数量,接着是这些路口的序号,序号按照升序排列。

第二行:跑道中“中间路口”的数量,接着是这些路口的序号,序号按照升序排列。


样例输入

1 2 -2
3 -2
3 -2
5 4 -2
6 4 -2
6 -2
7 8 -2
9 -2
5 9 -2
-2
-1

样例输出

2 3 6
13

首先我们可以枚举每一个点,看看删掉后能不能从起点走到终点,如果不可以,说明这个点是“不可避免的路口”

显而易见,如果一个点不是“不可避免的路口”,那么它不可能是“中间路口”

而一个点如果是“不可避免的路口”,那么就可以尝试从起点走到这个点,以及从这个点走到终点,如果路径有重合,则说明不是“中间路口”

这道题主要难在对细节的处理,可以给大家列举几个:

  1. 清空vis数组
  2. 中间路口并不算重合
  3. 题目说了不用枚举起点与终点
  4. 刚开始要把自己标记上(防止出现环)
#include<bits/stdc++.h>
using namespace std;
int n,cnt,tot;
const int N=55;
int g[N][N],vis1[N],vis2[N],bk[N],zj[N];

void dfs1(int x){
	vis1[x]=1;
	for(int i=0;i<=n;i++){
		if(!vis1[i]&&g[x][i]){
			dfs1(i);
		}
	}
}
void dfs2(int x){
	vis2[x]=1;
	for(int i=0;i<=n;i++){
		if(!vis2[i]&&g[x][i]){
			dfs2(i);
		}
	}
}
int main()
{
	while(true){
		int x;
		scanf("%d",&x);
		if(x==-1) break;
		if(x==-2) n++;
		else g[n][x]=1;
	}
	n--;
	for(int i=1;i<=n-1;i++){//第三个细节
		memset(vis1,0,sizeof(vis1));//第一个细节
		vis1[i]=1;//第四个细节
		dfs1(0);
		if(!vis1[n]){
			bk[++cnt]=i;
			memset(vis2,0,sizeof(vis2));
			vis2[i]=1;
			dfs2(i);
			int check=1;
			for(int j=0;j<=n;j++){
				if(vis1[j]&&vis2[j]&&j!=i){//j!=i处理第二个细节
					check=0;
					break;
				}
			}
			if(check) zj[++tot]=i;
		}
	}
	printf("%d ",cnt);
	for(int i=1;i<=cnt;i++) printf("%d ",bk[i]);
	printf("\n%d ",tot);
	for(int i=1;i<=tot;i++) printf("%d ",zj[i]);
}

标签:赛跑,终点,int,路口,跑道,街道,起点
From: https://www.cnblogs.com/HEIMOFA/p/17288207.html

相关文章

  • 模拟龟兔赛跑
    packageedu.wtbu;//模拟龟兔赛跑publicclassDemo01implementsRunnable{//胜利者privatestaticStringwinner;@Overridepublicvoidrun(){......
  • 澄江街道办配置AEE 执法记录仪,打造“智慧支撑+社区治理”新格局
    随着城市化的发展,人口不断向城市聚集,管理工作环境日渐复杂。为进一步提高街道办管理执法效率,规范街道办管理执法行为,澄江街道办为一线管理人员配置了AEEDSJ-K5高清红外现场......
  • 2023年3月最新全国省市区县和乡镇街道行政区划矢量边界坐标经纬度地图数据 shp geojso
    发现个可以免费下载全国 geojson 数据的网站,推荐一下。支持全国、省级、市级、区/县级、街道/乡镇级以及各级的联动数据,支持导入矢量地图渲染框架中使用,例如:D3、Echarts......
  • 多线程 龟兔赛跑案例
    packagecom.Java;publicclassRaceimplementsRunnable{//胜利者privatestaticStringwinner;@Overridepublicvoidrun(){for(inti=0......
  • 龟兔赛跑代码实现
    赛道相当于一个资源,乌龟和兔子相当于两个线程,多线程共用一个资源packagecom.demo01;publicclassRaceimplementsRunnable{privatestaticStringwinner;......
  • 广东电缆厂,闪电击中电缆,俄罗斯街道下起“电光雨”
    俄罗斯圣彼得堡市近日出现了一场奇观:当地政府时间——19日,圣彼得堡市遭受雷雨气温,一道闪电出乎意料击中无轨电车电缆,当即将电缆劈断、火花四散,这条路一瞬间被照亮,下起了电......
  • java从地址中解析省市区街道信息
    文章转载自: https://blog.csdn.net/renfng/article/details/94738164地址解析步骤如下:1、检查是否存在省份2、如果存在省份,将会检查省份是否明确标注省,自治区,市(直辖市),特......
  • 龟兔赛跑
        publicclassThreadDemo01implementsRunnable{privatestaticStringwinner;publicvoidrun(){for(inti=0;i<=100;i++)......
  • C/C++多线程实现龟兔赛跑
    题⽬:⻳兔赛跑跑道距离50⽶乌⻳(⼀个线程)每秒3⽶不睡觉;兔⼦(⼀个线程)每秒5⽶每跑15⽶睡2秒钟。请模拟⽐赛情况:#include<iostream>#include<thread>#include<......
  • 盖瑞特电动涡轮增压器:加速技术创新-从赛道到街道
    专业赛车在盖瑞特长达65年的创新历史中充当着前沿科技的孵化器。赛车运动像是一座实时测试实验室,不受限于预算、量产以及社会主流意识的束缚,它激发我们思考、探索、梦想、研......