CodeForces 954D-Fight Against Traffic(加边最短路)

题目链接:https://codeforces.com/problemset/problem/954/D
博客园食用链接:https://www.cnblogs.com/lonely-wind-/p/13451829.html

Little town Nsk consists of n junctions connected by m bidirectional roads. Each road connects two distinct junctions and no two roads connect the same pair of junctions. It is possible to get from any junction to any other junction by these roads. The distance between two junctions is equal to the minimum possible number of roads on a path between them.

In order to improve the transportation system, the city council asks mayor to build one new road. The problem is that the mayor has just bought a wonderful new car and he really enjoys a ride from his home, located near junction s to work located near junction t. Thus, he wants to build a new road in such a way that the distance between these two junctions won’t decrease.

You are assigned a task to compute the number of pairs of junctions that are not connected by the road, such that if the new road between these two junctions is built the distance between s and t won’t decrease.

Input
The firt line of the input contains integers n , m , s n, m, s n,m,s and t ( 2   ≤   n   ≤   1000 , 1   ≤   m   ≤   1000 , 1   ≤   s ,   t   ≤   n , s   ≠   t ) t (2 ≤ n ≤ 1000, 1 ≤ m ≤ 1000, 1 ≤ s, t ≤ n, s ≠ t) t(2n1000,1m1000,1s,tn,s=t) — the number of junctions and the number of roads in Nsk, as well as the indices of junctions where mayors home and work are located respectively. The i-th of the following m lines contains two integers u i u_ i ui and v i ( 1   ≤   u i ,   v i   ≤   n , u i   ≠   v i ) v_i (1 ≤ u_ i, v_ i ≤ n, u_i ≠ v i) vi(1ui,vin,ui=vi), meaning that this road connects junctions u i u _i ui and v i v_i vi directly. It is guaranteed that there is a path between any two junctions and no two roads connect the same pair of junctions.

Output
Print one integer — the number of pairs of junctions not connected by a direct road, such that building a road between these two junctions won’t decrease the distance between junctions s and t.

Examples
Input
5 4 1 5
1 2
2 3
3 4
4 5
Output
0

Input
5 4 3 5
1 2
2 3
3 4
4 5
Output
5

Input
5 6 1 5
1 2
1 3
1 4
4 5
3 5
2 5
Output
3

emmm,题目读的有点迷,不过只要注意到新加一个边和使得 s s s t t t的距离不减少(!!神奇的的要求),并且要求新加边的两个点直接之前不能有直达的边,这题就差不多读懂了

题目大意:给你一个图,n个点,m条边,起点和终点 s , t s,t s,t,然后你需要新加一条边在两个点之间,现在问你有多少种加法使得 d i s t ( s , t ) dist(s,t) dist(s,t)不减少。

看题目数据范围,比较小,那么我们可以直接枚举在那两个点之间建边,如果对于点 i , j i,j i,j建边了,若导致了最短路缩减,也就是 d i s ( s , i ) + d i s ( j , t ) < d i s ( s , t ) dis(s,i)+dis(j,t)<dis(s,t) dis(s,i)+dis(j,t)<dis(s,t) 或着 d i s ( s , j ) + d i s ( i , t ) < d i s ( s , t ) dis(s,j)+dis(i,t)<dis(s,t) dis(s,j)+dis(i,t)<dis(s,t)因为没法判断 i , j i,j i,j哪个点离哪个端点最近,所以要判断两次。

如果不减少的话也就是大于等于了,那么我们跑两次最短路就好了,一个对起点跑,一个对终点跑。然后枚举判断一下就好了。对于这种小数据,邻接矩阵他不香吗QAQ。

以下是AC代码:

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

const int mac=1e3+10;
const int inf=1e9+10;

int mp[mac][mac],diss[mac],n,dist[mac];
int vis[mac];

void dij(int s,int *dis)
{
	memset(vis,0,sizeof vis);
	for (int i=1; i<=n; i++) dis[i]=mp[s][i];
	dis[s]=0; vis[s]=1;
	for (int i=1; i<=n; i++){
		int minn=inf,k=0;
		for (int j=1; j<=n; j++)
			if (!vis[j] && dis[j]<minn)
				minn=dis[j],k=j;
		if (!k) break;
		vis[k]=1;
		for (int j=1; j<=n; j++)
			if (dis[k]+mp[k][j]<dis[j])
				dis[j]=dis[k]+mp[k][j];
	}
}

int main(int argc, char const *argv[])
{
	int m,s,t;
	scanf ("%d%d%d%d",&n,&m,&s,&t);
	for (int i=1; i<=n; i++)
		for (int j=1; j<=n; j++) mp[i][j]=inf;
	for (int i=1; i<=m; i++){
		int u,v;
		scanf ("%d%d",&u,&v);
		mp[u][v]=mp[v][u]=1;
	}
	for (int i=1; i<=n; i++) mp[i][i]=0;
	dij(s,diss); dij(t,dist);
	int ans=0;
	for (int i=1; i<=n; i++){
		for (int j=i+1; j<=n; j++){
			if (mp[i][j]!=inf) continue;
			if (diss[i]+dist[j]+1>=diss[t] && dist[i]+diss[j]+1>=diss[t]) 
				ans++;
		}
	}
	printf("%d\n",ans);
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
CodeForces - 616D是一个关于找到一个序列中最长的第k好子段的起始位置和结束位置的问题。给定一个长度为n的序列和一个整数k,需要找到一个子段,该子段中不超过k个不同的数字。题目要求输出这个序列最长的第k好子段的起始位置和终止位置。 解决这个问题的方法有两种。第一种方法是使用尺取算法,通过维护一个滑动窗口来记录\[l,r\]中不同数的个数。每次如果这个数小于k,就将r向右移动一位;如果已经大于k,则将l向右移动一位,直到个数不大于k。每次更新完r之后,判断r-l+1是否比已有答案更优来更新答案。这种方法的时间复杂度为O(n)。 第二种方法是使用枚举r和双指针的方法。通过维护一个最小的l,满足\[l,r\]最多只有k种数。使用一个map来判断数的种类。遍历序列,如果当前数字在map中不存在,则将种类数sum加一;如果sum大于k,则将l向右移动一位,直到sum不大于k。每次更新完r之后,判断i-l+1是否大于等于y-x+1来更新答案。这种方法的时间复杂度为O(n)。 以上是两种解决CodeForces - 616D问题的方法。具体的代码实现可以参考引用\[1\]和引用\[2\]中的代码。 #### 引用[.reference_title] - *1* [CodeForces 616 D. Longest k-Good Segment(尺取)](https://blog.csdn.net/V5ZSQ/article/details/50750827)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Codeforces616 D. Longest k-Good Segment(双指针+map)](https://blog.csdn.net/weixin_44178736/article/details/114328999)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值