route(2018.10.24)
建出最短路图之后\(topsort\)即可。
具体思路:
先用\(dijkstra\)算法在原图中跑出\(1\)号点到\(i\)号节点的最短距离\(dist_1(i)\),将所有边反向后用\(dijkstra\)算法求出\(i\)号点到\(2\)号点的最短距离\(dist_2(i)\);
再沿着最短路径找到从\(1\)号点到\(i\)号点的方案数\(f(i)\),以及\(i\)号点到\(2\)号点的方案数\(g(i)\);
如果一条起点为\(a_i\),终点为\(b_i\),长度为\(c_i\)的边满足\(f(a_i)*g(b_i)==f(2)\)且\(dist_1(i)+c_i+dist_2(i)==dist_1(2)\)则该边是原图中从\(1\)号点到\(2\)号的最短路径的必经之路。
将第i条边反向后对该边进行分类讨论
\(1.dist_2(ai)+dist_1(bi)+ci < dist_1(2)\) 则最短路径的长度变短了
\(2.dist_2(ai)+dist_1(bi)+ci == dist_1(2)\) ,则最短路径的长度并没有发生变化
\(3.dist_2(ai)+dist_1(bi)+ci > dist_1(2)\)
\(a.\)如果该边不是原图最短路径的必经之路,则最短路径的长度并没有发生变化
\(b.\)如果该边是原图最短路径的必经之路,则最短路径的长度变长了或者新图中不存在从\(1\)号点到\(2\)号点的路径
(当不存在从\(1\)号点到\(i\)号点的路径时,\(dist_1(i)\)为极大值,\(dist_2(i)\)同理)
时间复杂度\(O((m+n)log(n))\),但当路径数量过多时,\(f\)值和\(g\)值会超出\(int\)的储存范围
期望\(100\)分解法:
在上述求必经之路的时候用拓扑序或者其他方法求在最短路径构成的图上的桥
但是我太傻逼了,唉!
代码:
#include<cstdio>
#include<queue>
#include<cstring>
using namespace std;
int n,m,x[100001],y[100001],cnt,ans[100001],z[100001],dis[100001],dis1[100001],pre[100001],nxt[100001],h[100001],v[100001],c[100001],sum,in[100001],pree[100001],nxtt[100001],hh[100001],vv[100001],f[100001],g[100001];
struct oo{int x,y;};bool vis[100001];
bool operator<(oo a,oo b){return a.x>b.x;}
priority_queue<oo>q;
void add(int x,int y,int z){pre[++cnt]=y;nxt[cnt]=h[x];h[x]=cnt;v[cnt]=z;}
void ins(int x,int y,int z){pree[++cnt]=y;nxtt[cnt]=hh[x];hh[x]=cnt;vv[cnt]=z;}
void dijkstra(int x,int *dis,int id)
{
memset(vis,0,sizeof vis);
q.push((oo){0,x});dis[x]=0;
while(!q.empty())
{
oo x=q.top();q.pop();
if(vis[x.y])continue;vis[x.y]=1;
for(int i=h[x.y];i;i=nxt[i])
if(dis[pre[i]]>dis[x.y]+v[i])
{
dis[pre[i]]=dis[x.y]+v[i];
q.push((oo){dis[pre[i]],pre[i]});
}
}
}
void topsort(int x,int *f)
{
queue<int>que;
que.push(x);f[x]=1;
while(!que.empty())
{
int now=que.front();que.pop();
for(int i=hh[now];i;i=nxtt[i])
{
f[pree[i]]+=f[now];
if(!(--in[pree[i]]))que.push(pree[i]);
}
}
}
int main()
{
// freopen("route.in","r",stdin);
// freopen("route.out","w",stdout);
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)scanf("%d%d%d",&x[i],&y[i],&z[i]),add(y[i],x[i],z[i]);
memset(dis1,63,sizeof dis1);memset(dis,63,sizeof dis);
dijkstra(2,dis1,0);
memset(h,0,sizeof h);cnt=0;
for(int i=1;i<=m;i++)add(x[i],y[i],z[i]);
dijkstra(1,dis,1);cnt=0;
memset(vis,0,sizeof vis);
for(int i=1;i<=n;i++)
for(int j=h[i];j;j=nxt[j])
if(dis[i]+v[j]+dis1[pre[j]]==dis[2])vis[j]=1,ins(i,pre[j],v[j]),x[cnt]=i,y[cnt]=pre[j],z[cnt]=v[j],in[pre[j]]++;
topsort(1,f);
memset(hh,0,sizeof hh);memset(in,0,sizeof in);int sum=cnt;cnt=0;
for(int i=1;i<=sum;i++)ins(y[i],x[i],z[i]),in[x[i]]++;
topsort(2,g);
for(int i=1;i<=n;i++)
for(int j=h[i];j;j=nxt[j])
{
if(vis[j]){if(f[i]*g[pre[j]]==f[2])ans[j]=1;}
else if(dis1[i]+v[j]+dis[pre[j]]<dis[2])ans[j]=-1;
}
for(int i=1;i<=m;i++)printf("%d\n",ans[i]);
}
route(2018.10.24)的更多相关文章
- python中使用Opencv进行车牌号检测——2018.10.24
初学Python.Opencv,想用它做个实例解决车牌号检测. 车牌号检测需要分为四个部分:1.车辆图像获取.2.车牌定位.3.车牌字符分割和4.车牌字符识别 在百度查到了车牌识别部分车牌定位和车牌字 ...
- 2018.10.24 NOIP2018模拟赛 解题报告
得分: \(100+0+100=200\)(\(T2\)悲惨爆\(0\)) \(P.S.\)由于原题是图片,所以我没有上传题目描述,只有数据. \(T1\):query(点此看题面) 熟悉主席树的人都 ...
- 课堂笔记及知识点----树(2018/10/24(pm))
树 概念:由一个或多个(n≥0)结点组成的有限集合 T, 有且仅有一个结点称为根( root), 当 n>1时,其余的结点分为 m(m≥0)个互不相交的有限集合 T1,T2, …, Tm.每个集 ...
- 课堂笔记及知识点----栈和队列(2018/10/24(am))
栈: Stack<int> xt=new Stack<int>() ; 先进后出,后进先出,水杯结构,顺序表类似 常用方法: .pop---->出栈,弹栈 ...
- 2018.10.24 bzoj3195: [Jxoi2012]奇怪的道路(状压dp)
传送门 f[i][j][k]f[i][j][k]f[i][j][k]表示前iii个点连了jjj条边,第i−K+1i-K+1i−K+1~iii个点连边数的奇偶性为kkk时的方案数. 转移规定只能从后向前 ...
- 2018.10.24 bzoj2064: 分裂(状压dp)
传送门 状压dp好题. 考虑对于两个给出的集合. 如果没有两个元素和相等的子集,那么只能全部拼起来之后再拆开,一共需要n1+n2−2n1+n2-2n1+n2−2. 如果有呢? 那么对于没有的就是子问题 ...
- 2018.10.24 NOIP模拟 小 C 的宿舍(分治)
传送门 分治妙题. 没有这道题的暴力分今天又垫底了啊233 由于用了分治的方法,我们只用考虑左区间对右区间的贡献以及右区间对左区间的贡献. 可以发现如果从中点开始向两边递推最小值并用这个区间最小值来推 ...
- 2018.10.24 NOIP模拟 小 C 的序列(链表+数论)
传送门 考虑到a[l],gcd(a[l],a[l+1]),gcd(a[l],a[l+1],a[l+2])....gcd(a[l]...a[r])a[l],gcd(a[l],a[l+1]),gcd(a[ ...
- 2018.10.24 NOIP模拟 小 C 的数组(二分+dp)
传送门 考试自己yyyyyy的乱搞的没过大样例二分+dp二分+dp二分+dp过了606060把我自己都吓到了! 这么说来乱搞跟被卡常的正解比只少101010分? 那我考场不打其他暴力想正解血亏啊. 正 ...
随机推荐
- 1 Maven简介
一.构建(build) 清理.编译.测试.打包.部署等一系列操作. 二.maven介绍: maven是一个强大的构建工具,能够帮助我们自动化构建过程:从清理(clean).编译(com ...
- Windows Power Shell
Windows PowerShell 是一种命令行外壳程序和脚本环境,使命令行用户和脚本编写者可以利用 .NET Framework的强大功能. 它引入了许多非常有用的新概念,从而进一步扩展了您在 W ...
- ansible-playbook unarchive模块
先 进行 pause模块的 记录: pause 在playbook执行的过程中暂停一定时间或者提示用户进行某些操作 常用参数: minutes:暂停多少分钟 seconds:暂停多少秒 prompt ...
- 关于URL编码的一些结论
转载自:http://www.ruanyifeng.com/blog/2010/02/url_encoding.html与http://www.ruanyifeng.com/blog/2007/10/ ...
- 【转载】如何在 C#中访问 JavaScript函数?
如何在 C#中访问 JavaScript函数? 时间:13-10-17 栏目:Unity3D教程 作者:zqcyou 评论:0 如何在 C#中访问 JavaScript函数?答案如下:c#代码 ...
- 并不对劲的bzoj5322:loj2543:p4561:[JXOI2018]排序问题
题目大意 \(T\)(\(T\leq10^5\))组询问 每次给出\(n,m,l,r\),和\(n\)个数\(a_1,a_2,...,a_n\),要找出\(m\)个可重复的在区间\([l,r]\)的数 ...
- BZOJ_2369_区间_决策单调性
BZOJ_2369_区间_决策单调性 Description 对于一个区间集合 {A1,A2……Ak}(K>1,Ai不等于Aj(i不等于J),定义其权值 S=|A1∪A2∪……AK|*|A1 ...
- C语言解释器的实现--存储结构(一)
目录: 1. 内存池 2. 栈 3. Hash表 1.内存池 在一些小的程序里,没什么必要添加内存管理模块在里面.但是对于比较复杂的代码,如果需要很多的内存操作,那么加入自己的内存管理是有必要的.至 ...
- 配置tomcat https的步骤
keytool -genkey -v -alias tomcat -keyalg RSA -keystore D:\tomcat.keystore -validity 36500 keytool -g ...
- HDU2222(AC自动机入门题)
Keywords Search Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others ...