CJOJ 免费航班

Posted qt666

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CJOJ 免费航班相关的知识,希望对你有一定的参考价值。

Description

小Z在MOI比赛中获得了大奖,奖品是一张特殊的机 票。使用这张机票,可以在任意一个国家内的任意城市之间的免费飞行,只有跨国飞行时才会有额外的费用。小Z获得了一张地图,地图上有城市之间的飞机航班和 费用。已知从每个城市出发能到达所有城市,两个城市之间可能有不止一个航班。一个国家内的每两个城市之间一定有不止一条飞行路线,而两个国家的城市之间只 有一条飞行路线。小Z想知道,从每个城市出发到额外费用最大的城市,以便估算出出行的费用,请你帮助他。当然,你不能通过乘坐多次一个航班增加额外费用, 也就是必须沿费用最少的路线飞行。

Input

第一行,两个整数N,M,表示地图上有N个城市,M条航线。
接下来M行,每行三个整数a,b,c,表示城市a,b之间有一条费用为c的航线。

Output

共N行,第i行为从城市i出发到达每个城市额外费用的最大值。

Sample Input

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

Sample Output

4
4
4
6
7
7

Hint

样例说明
有四个国家,包含的城市分别为 {1,2,3},{4},{5},{6}。从城市1出发到达城市6,乘坐(1,3)(3,6)两个航班费用最大,(1,3)在国内为免费航班,(3,6)的费用为4,所以从1出发的最大费用为4。

数据规模
对于30%的数据 1<=N<=1000,1<=M<=1000
对于100%的数据 1<=N<=20000,1<=M<=200000

Source

动态规划 ,连通性

 

根据题目描述,每个国家是一个双连通分量,把每个双连通分量缩点后,原图变为一棵树.

相当于是求树上每个点在树上的最长路.

所有点的树上最长路可以通过两边dfs进行DP;

第一遍:求出每个点只到他子树内部的最长路和次长路

第二遍:每个点再由他父亲来更新往改点的子树外走的最长路,因为该点到子树外面的路必经过他爸爸

具体实现就是最长路和次长路转化,转移画画图就好了

双连通缩点的话就是把桥标记后在dfs一遍

#include<cstdio>
#include<algorithm>
#include<iostream>
#include<cstring>
#include<vector>
#define RG register
using namespace std;
typedef long long ll;
const int N=20050;
int gi(){
  int x=0;
  char ch=getchar();
  while(ch<‘0‘||ch>‘9‘) ch=getchar();
  while(ch>=‘0‘&&ch<=‘9‘) x=x*10+ch-‘0‘,ch=getchar();
  return x;
}
int head[N],nxt[N*20],to[N*20],dfn[N],low[N],vis[N],zhan[N],w[N*20],cnt=1,tot,sum,top,pd[N*20];
vector<int>q[N],p[N],W[N];
int dis[N][2],vis2[N],fr[N];
void tarjan(int x,int fa){
  dfn[x]=low[x]=++sum;
  for(RG int i=head[x];i;i=nxt[i]){
      int y=to[i];
      if(!dfn[y]){
	  tarjan(y,x);
	  low[x]=min(low[x],low[y]);
	  if(low[y]>dfn[x]) pd[i]=pd[i^1]=1;
	}
      else if(y!=fa) low[x]=min(low[x],dfn[y]);
    }
}
void dfs3(int x,int gg){
  fr[x]=gg,vis[x]=1;
  q[gg].push_back(x);
  for(RG int i=head[x];i;i=nxt[i]){
      if(!pd[i]&&!vis[to[i]]) dfs3(to[i],tot);
    }
}
void dfs1(int x){
    vis2[x]=1;
    for(RG int i=0;i<p[x].size();i++){
	int y=p[x][i],w=W[x][i];
	if(!vis2[y]){
	    dfs1(y);
	    if(dis[y][1]+w>=dis[x][1]){
		dis[x][0]=dis[x][1];
		dis[x][1]=dis[y][1]+w;
	    }
	    else dis[x][0]=max(dis[x][0],dis[y][1]+w);
	}
    }
}
void dfs2(int x){
    vis2[x]=1;
    for(RG int i=0;i<p[x].size();i++){
	int y=p[x][i],w=W[x][i];
	if(!vis2[y]){
	    if(dis[y][1]+w==dis[x][1]){
		dis[y][0]=max(dis[y][0],min(dis[x][1]-w,dis[x][0]+w));
		dis[y][1]=max(dis[x][1]-w,dis[x][0]+w);
	    }
	    else{
		dis[y][0]=max(dis[y][0],max(dis[x][0]+w,min(dis[y][1],dis[x][1]+w)));
		dis[y][1]=max(dis[y][1],dis[x][1]+w);
	    }
	    dfs2(y);
	}
    }
}
int main(){
  int n,m,x,w1,y;
  cnt=1;n=gi(),m=gi();
  for(RG int i=1;i<=m;i++){
      x=gi(),y=gi(),w1=gi();
      to[++cnt]=y,w[cnt]=w1,nxt[cnt]=head[x],head[x]=cnt;
      to[++cnt]=x,w[cnt]=w1,nxt[cnt]=head[y],head[y]=cnt;
    }
  tarjan(1,1);
  for(int i=1;i<=n;i++) if(!vis[i]) dfs3(i,++tot);
  for(RG int i=1;i<=tot;i++)
    for(RG int j=0;j<q[i].size();j++)
      for(RG int k=head[q[i][j]];k;k=nxt[k]){
	  int y=fr[to[k]],w1=w[k];
	  if(y!=i){
	      p[i].push_back(y);
	      W[i].push_back(w1);
	    }
	}
  dfs1(1);for(RG int i=1;i<=tot;i++) vis2[i]=0;
  dfs2(1);for(RG int i=1;i<=n;i++) printf("%d\n",dis[fr[i]][1]);
  return 0;
}

 

以上是关于CJOJ 免费航班的主要内容,如果未能解决你的问题,请参考以下文章

「CJOJ2574」Lucky Transformati

CJOJ1857 -PG图

软件开发 [CJOJ 1101] [NOIP 模拟]

(CJOJ)P2634 - NOIP2017模拟失格

「CJOJ2736」「POJ1014」大理石分割

钉钉杯大学生大数据挑战赛初赛B 航班数据分析与预测 Python代码实现Baseline