题目描述

给出一个n个点m条边的无向图,n个点的编号从1~n,定义源点为1。定义最短路树如下:从源点1经过边集T到任意一点i有且仅有一条路径,且这条路径是整个图1到i的最短路径,边集T构成最短路树。

给出最短路树,求对于除了源点1外的每个点i,求最短路,要求不经过给出的最短路树上的1到i的路径的最后一条边。

输入

第一行包含两个数n和m,表示图中有n个点和m条边。

接下来m行,每行有四个数ai,bi,li,ti,表示图中第i条边连接ai和bi权值为li,ti为1表示这条边是最短路树上的边,ti为0表示不是最短路树上的边。

输出

输出n-1个数,第i个数表示从1到i+1的要求的最短路。无法到达输出-1。

样例输入

5 9
3 1 3 1
1 4 2 1
2 1 6 0
2 3 4 0
5 2 3 0
3 2 2 1
5 3 1 1
3 5 2 0
4 5 4 0

样例输出

6 7 8 5

提示

对于30%的数据,n≤100,m≤2000
对于100%的数据,n≤4000,m≤100000,1≤li≤100000

分析:

考虑一个点的答案是怎么样的:从$1$到$i$的答案,应该是从$1$沿着树边走,然后经过一条非树边走到以$i$为根的子树中,在沿着树边向上走到$i$。画图可以证明走两条非树边一定更不优。$p[i][j]$表示$i->j$的只经过一条两端在$i$和$j$的子树非树边最短路,$dis[i]$表示$1->i$的最短路,所以$ans[i]=min(p[x][i]+dis[x])$,$x$是$i$子树中的一点。

那么只要处理$dis[i]$和$p[i][j]$,$dis[i]$只要求树上距离即可,$p[i][j]$的计算类似更新答案,$dfs$过程中枚举$1~n$个点,然后用用儿子节点更新当前节点即可。

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#define M 200005
#define N 4005
using namespace std;
int n,m,dis[N],p[N][N],ans[N],pre[N],las[N],ind;
int hd[N],to[M],nx[M],w[M],cnt;
inline void add(int u,int v,int c){
to[++cnt]=v;w[cnt]=c;nx[cnt]=hd[u];hd[u]=cnt;
}
inline void dfs1(int u,int f){
pre[u]=++ind;
for(int i=hd[u];i;i=nx[i]){
int v=to[i];
if(v!=f){
dis[v]=dis[u]+w[i];
dfs1(v,u);
}
}
las[u]=ind;
}
inline void dfs2(int u,int f){
for(int i=hd[u];i;i=nx[i]){
int v=to[i];
if(v!=f){
dfs2(v,u);
for(int j=1;j<=n;j++){
if(pre[j]>=pre[u]&&las[j]<=las[u]) continue;
p[u][j]=min(p[u][j],p[v][j]+w[i]);
}
}
}
for(int i=1;i<=n;i++){
if(pre[i]>=pre[u]&&las[i]<=las[u]) continue;
ans[u]=min(ans[u],p[u][i]+dis[i]);
}
}
int main(){
scanf("%d%d",&n,&m);
memset(p,123,sizeof p);
for(int i=1;i<=m;i++){
int a,b,l,t;
scanf("%d%d%d%d",&a,&b,&l,&t);
if(t){add(a,b,l);add(b,a,l);}
else p[a][b]=p[b][a]=min(p[a][b],l);
}
memset(ans,123,sizeof ans);
dfs1(1,0);dfs2(1,0);
for(int i=2;i<=n;i++)
if(ans[i]==ans[0]) puts("-1");
else printf("%d\n",ans[i]);
return 0;
}

总结:

图论等,尤其是图论题。很多时候要考虑答案长什么样,什么情况下最优,当有了特殊性质时,题目会更好做。

如果要证明一些性质,可以先假设一些条件,然后用反证法。例如:假设最优答案在(……)上,如果有更优的答案,那么(……)就不是(……)。之类的……

BZOJ 3694&&DTOJ 1972: 最短路的更多相关文章

  1. Bzoj 3694: 最短路 树链剖分

    3694: 最短路 Time Limit: 5 Sec  Memory Limit: 256 MBSubmit: 67  Solved: 34[Submit][Status][Discuss] Des ...

  2. BZOJ 3694 最短路

    233333想简单了.... 题解参见http://hzwer.com/3710.html #include<iostream> #include<cstdio> #inclu ...

  3. bzoj 1295 最长距离 - 最短路

    Description windy有一块矩形土地,被分为 N*M 块 1*1 的小格子. 有的格子含有障碍物. 如果从格子A可以走到格子B,那么两个格子的距离就为两个格子中心的欧几里德距离. 如果从格 ...

  4. BZOJ 2763 分层图最短路

    突然发现我不会分层图最短路,写一发. 就是同层中用双向边相连,用单向边连下一层 #include <cstdio> #include <algorithm> #include ...

  5. BZOJ 2763: [JLOI2011]飞行路线 最短路

    2763: [JLOI2011]飞行路线 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  6. BZOJ 2750: [HAOI2012]Road( 最短路 )

    对于每个点都跑最短路, 然后我们得到了个DAG, 在这DAG上更新每条边的答案. 考虑e(u, v)∈DAG对答案的贡献:  假设从S到u得路径数为A[u], 从v出发到达任意点的路径数为B[v], ...

  7. BZOJ.2069.[POI2004]ZAW(最短路Dijkstra 按位划分)

    题目链接 \(Description\) 给定一张带权图(边是双向的,但不同方向长度不同).求从1出发,至少经过除1外的一个点,再回到1的最短路.点和边不能重复经过. \(n\leq5000,m\le ...

  8. BZOJ 1003 - 物流运输 - [最短路+dp]

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1003 Time Limit: 10 Sec Memory Limit: 162 MB D ...

  9. BZOJ 1922--大陆争霸(最短路)

    1922: [Sdoi2010]大陆争霸 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 2113  Solved: 947[Submit][Status ...

随机推荐

  1. 【Java虚拟机4】Java内存模型(硬件层面的并发优化基础知识--缓存一致性问题)

    前言 今天学习了Java内存模型第一课的视频,讲了硬件层面的知识,还是和大学时一样,醍醐灌顶.老师讲得太好了. Java内存模型,感觉以前学得比较抽象.很繁杂,抽象. 这次试着系统一点跟着2个老师学习 ...

  2. 新手小白在github上部署一个项目

    新手小白在github上部署一个项目 一. 注册github账号 github地址:https://www.github.com/ 二.下载安装Git 地址:https://git-scm.com/d ...

  3. MySQL:提高笔记-2

    MySQL:提高笔记-2 学完基础的语法后,进一步对 MySQL 进行学习,第一篇为:MySQL:提高笔记-1,这是第二篇内容 说明:这是根据 bilibili 上 黑马程序员 的课程 mysql入门 ...

  4. BUAA2020软工作业——提问回顾与个人总结

    项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 提问回顾与个人总结 我在这个课程的目标是 进一步提高自己的编码能力,工程能力 这个作业在哪个具体方 ...

  5. Spring父子上下文的使用案例

    Spring父子上下文的使用案例 一.背景 二.需求 三.实现步骤 1.基础代码编写 2.测试结果 四.小彩蛋 五.完整代码 一.背景 最近在看在使用Spring Cloud的时候发现,当我们通过Fe ...

  6. 关于qmake的install

    在pro的构建系统中可以设置INSTALLS变量,在make命令之后,执行make install命令触发,将想要的资源拷贝到相应的目录,参考qwt的构建体系,在qwt.pro末尾有这么几句 qwts ...

  7. 零基础入门必备的Linux命令和C语言基础

    文件和目录(底部有视频资料) cd /home 进入 '/ home' 目录' cd - 返回上一级目录 cd -/- 返回上两级目录 cd 进入个人的主目录 cd ~user1 进入个人的主目录 c ...

  8. 原串反转 牛客网 程序员面试金典 C++ Python

    原串反转 牛客网 程序员面试金典 C++ Python 题目描述 请实现一个算法,在不使用额外数据结构和储存空间的情况下,翻转一个给定的字符串(可以使用单个过程变量). 给定一个string iniS ...

  9. JavaScript复习 1

    概括及使用方法: JavaScript编写规范 一般放在<head>-</head>中间 逐行被执行,越短越好 大小写敏感 语句是基本单位 通常以分号表示语句结束 多行语句可以 ...

  10. BugKu之备份是个好习惯

    题目:备份是个好习惯 思路分析 打开题目,看到一个字符串. 联系到题目,就猜到肯定是源代码泄露,用工具扫一下,发现了index.php.bak,验证了我的猜想,下载下来看看. <?php /** ...