道路拆除。。

[CSP-J2019 江西] 道路拆除

题目描述

A 国有 n 座城市,从 1 \sim n 编号。 1 号城市是 A 国的首都。城市间由 m 条双向道路连通,通过每一条道路所花费的时间均为 1 单位时间。

现在 A 国打算拆除一些不实用的道路以减小维护的开支,但 A 国也需要保证主要线路不受影响。因此 A 国希望道路拆除完毕后,利用剩余未被拆除的道路,从 A 国首都出发,能到达 s_1 号与 s_2 号城市,且所要花费的最短时间分别不超过 t_1t_2 (注意这是两个独立的条件,互相之间没有关联,即不需要先到 s_1 再到 s_2 )。

A 国想请你帮他们算算,在满足上述条件的情况下,他们最多能拆除多少条道路。 若上述条件永远无法满足,则输出 -1

输入格式

第一行两个正整数 n,m ,表示城市数与道路数。

接下来 m 行,每行两个正整数 x,y ,表示一条连接 x 号点与 y 号点的道路。

最后一行四个整数,分别为 s_1,t_1,s_2,t_2

输出格式

仅一行一个整数,表示答案。

样例 #1

样例输入 #1

5 6
1 2
2 3
1 3
3 4
4 5
3 5
5 3 4 3

样例输出 #1

3

样例 #2

样例输入 #2

3 2
1 2
2 3
2 2 3 1

样例输出 #2

-1

提示

【数据范围】
对于 30\% 的数据,n ,m15
另有 20\% 的数据,n ≤ 100,m = n - 1
另有 30\% 的数据,s_1 = s_2
对于 100\% 的数据,2n,m3000,1 ≤ x,y ≤ n,2 ≤ s_1,s_2 ≤ n,0 ≤ t_1,t_2 ≤ n

【样例 1 解释】
拆除 (1,2),(2,3),(3,4) 三条边。
注意:不需要令首都与除了 s_1,s_2 外的点在拆除之后依然连通。

【样例 2 解释】
即使一条边都不拆除,首都到 3 号点的最短时间也都达到了 2 单位时间。

思路

看到这一道题的第一想法就是跑最短路。可是仔细想想就发现,由于重合的路径只算一遍,所以导致两条最短路不一定是最优解。
接着,看到数据范围中的 m≤3000
m≤3000 告诉我们这个无向图是稀疏图。也就是说,从 11 到 s1,s2,s1,s2 的简单路径(重复点或边没有意义)总数不会很多。因此,我们就可以穷举 (1,s1),(1,s2)
(1,s1),(1,s2) 的所有简单路径,求最小经过的边即可。
只要加上以下的基础剪枝即可:
如果经过的边数已经超过目前最小值,返回。
如果路径长度已经超过 t1,t1 or t2 t2,返回。
部分代码

int spfa(int x , int dis[]){
	queue<int> q;
	memset(vis , 0 , sizeof(vis));
	q.push(x);
	dis[x] = 0;
	vis[x] = 1;
	while(!q.empty()){
		int xx = q.front();
		q.pop();
		vis[xx] = 0;
		for(int i = 0; i < e[xx].size(); i++){
			int y = e[xx][i];
			if(dis[y] > dis[xx] + 1){
				dis[y] = dis[xx] + 1;
				if(!vis[y]){
					vis[y] = 1;
					q.push(y);
				}
			}
		}
	}
}