luogu4061 大吉大利,晚上吃鸡!
- 最短路径\(dag\),一道好题。
- 题目大意:求一张图中满足下列要求的点对\((i,j)\)数量:
- 所有最短路径必定会经过 \(i\) 点和 \(j\) 点中的任意一点。
- 不存在一条最短路同时经过 \(i\) 点和 \(j\) 点。
- 考虑这两个限制是啥。
- 首先所有最短路要么经过\(i\)点,要么经过\(j\)点,不存在两个都不经过或者都经过的。
- 所以经过\(i\)点的最短路方案加上经过\(j\)点的最短路方案不存在交集。
- 那么考虑怎么求最短路方案,先做一边\(dij\),然后在\(tag\)上\(dp\)方案数即可,正反都做一遍。
- 那么有\(f_{i,0}\)表示从\(s\)到\(i\)正向最短路方案数,
- 经过\(i\)点的最短路方案设为\(F_{i}\),有$${F_{i}=f_{i,0}*f_{i,1}}$$
- 所以满足要求的\(i,j\)可写为:$$F_i+F_j=F_t$$。
- 这个满足了还不够,因为这个只是\(i,j\)的必要条件,还不是充分条件。
- 关键在于所有的路径不能同时经过这两个点。
- 设\(G_{i,0}\)表示从\(s\)点到\(i\)点可能经过的点集,\(bitset\)实现,\(G_{i,1}\)同理。
- 转移拓扑排序实现即可。
- 所以我们对于每一个\(i\),想知道\(F_{t}-F_{i}=F_{j}\)中,\(j\)的点数,减去\(i\)可以到达的点数。
- 维护一个\(map\),记录\(F_{j}\)这个集合中的点。
- 那么对于一个\(i\),他能产生的贡献就是集合$$S=G_{F_t-F_i}\ and\ G_{i,0}\ and \ G_{i,1}$$
- 的\(1\)的个数。
- 空间\(1GB\)就可以过
// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define R register int
#define ll long long
using namespace std;
const int N=50001;
const int M=100001;
const ll inf=1e18;
const int mod=1000000009;
int n,m,s,u,v,x,t,ans,vis[N];
int cnt,nt[M],to[M],hd[N],w[M];
int du[N];
ll Dis[2][N],f[2][N],F[N];
void link(R f,R t,R d){nt[++cnt]=hd[f],to[cnt]=t,w[cnt]=x,hd[f]=cnt;}
bitset<50001>G[2][N];
map<ll,bitset<50001> >Ms;
int gi(){
R x=0,k=1;char c=getchar();
while((c<'0'||c>'9')&&c!='-')c=getchar();
if(c=='-')k=-1,c=getchar();
while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+c-'0',c=getchar();
return x*k;
}
struct ip{int id;ll val;};
int operator < (ip x,ip y){return x.val>y.val;}
priority_queue<ip>Q;
void dij(R op){
while(!Q.empty())Q.pop();
for(R i=1;i<=n;++i)vis[i]=0,Dis[op][i]=inf;
if(op)Dis[1][t]=0,Q.push((ip){t,0}),f[op][t]=1;
else Dis[0][s]=0,Q.push((ip){s,0}),f[op][s]=1;
while(!Q.empty()){
ip s=Q.top();Q.pop();
if(vis[s.id])continue;
R i=s.id;vis[i]=1;
for(R k=hd[i];k;k=nt[k]){
if(Dis[op][to[k]]>Dis[op][i]+w[k]){
Dis[op][to[k]]=Dis[op][i]+w[k];
f[op][to[k]]=f[op][i],Q.push((ip){to[k],Dis[op][to[k]]});
}
else if(Dis[op][to[k]]==Dis[op][i]+w[k])
f[op][to[k]]+=f[op][i];
}
}
}
queue<int>til;
void top(R op){
for(R i=1;i<=n;++i)
for(R k=hd[i];k;k=nt[k])
if(Dis[op][i]+w[k]+Dis[!op][to[k]]==Dis[0][t])
du[to[k]]++;
for(R i=1;i<=n;++i){
G[op][i].set(),G[op][i][0]=G[op][i][i]=0;
if(!du[i])til.push(i);
}
while(!til.empty()){
R i=til.front();til.pop();
for(R k=hd[i];k;k=nt[k])
if(Dis[op][i]+w[k]+Dis[!op][to[k]]==Dis[0][t]){
du[to[k]]--,G[op][to[k]]&=G[op][i];
if(!du[k])til.push(to[k]);
}
}
}
int main(){
n=gi(),m=gi(),s=gi(),t=gi();
for(R i=1;i<=m;++i)
u=gi(),v=gi(),x=gi(),link(u,v,x),link(v,u,x);
dij(0),dij(1);
if(!f[0][t])return printf("%lld",1ll*n*(n-1)/2),0;
for(R i=1;i<=n;++i)
if(Dis[0][i]+Dis[1][i]==Dis[0][t])
F[i]=f[0][i]*f[1][i];
top(0),top(1);
for(R i=1;i<=n;++i)Ms[F[i]].set(i);
for(R i=1;i<=n;++i)
ans+=(Ms[F[t]-F[i]]&G[0][i]&G[1][i]).count();
cout<<(ans>>1);
return 0;
}
luogu4061 大吉大利,晚上吃鸡!的更多相关文章
- GMA Round 1 大吉大利,晚上吃鸡
传送门 大吉大利,晚上吃鸡 新年走亲访友能干点啥呢,咱开黑吃鸡吧. 这里有32个人,每个人都可能想玩或者不想玩,这样子一共有$2^{32}$种可能.而要开黑当然得4人4人组一队(四人模式),所以说如果 ...
- [BZOJ5109]大吉大利,晚上吃鸡!
[BZOJ5109]大吉大利,晚上吃鸡! 题目大意: 一张\(n(n\le5\times10^4)\)个点\(m(m\le5\times10^4)\)条边的无向图,节点编号为\(1\)到\(n\),边 ...
- 【BZOJ5109】[CodePlus 2017]大吉大利,晚上吃鸡! 最短路+拓扑排序+DP
[BZOJ5109][CodePlus 2017]大吉大利,晚上吃鸡! Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏 ...
- bzoj5109: [CodePlus 2017]大吉大利,晚上吃鸡!
Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏中,皮皮 和毛毛最喜欢做的事情就是堵桥,每每有一个好时机都能收到不少的快 ...
- 「CodePlus 2017 11 月赛」大吉大利,晚上吃鸡!(dij+bitset)
从S出发跑dij,从T出发跑dij,顺便最短路计数. 令$F(x)$为$S$到$T$最短路经过$x$的方案数,显然这个是可以用$S$到$x$的方案数乘$T$到$x$的方案数来得到. 然后第一个条件就变 ...
- BZOJ5109 CodePlus 2017大吉大利,晚上吃鸡!(最短路+拓扑排序+bitset)
首先跑正反两遍dij求由起点/终点到某点的最短路条数,这样条件一就转化为f(S,A)*f(T,A)+f(S,B)*f(T,B)=f(S,T).同时建出最短路DAG,这样图中任何一条S到T的路径都是最短 ...
- 「CodePlus 2017 11 月赛」大吉大利,晚上吃鸡!
n<=50000,m<=50000的图,给s和t,问有多少点对$(a,b)$满足 嗯. 不会. 首先最短路DAG造出来,然后两个条件转述一下:条件一,$N_a$表示从s到t经过a的路径,$ ...
- [Code+#1]大吉大利,晚上吃鸡!
输入输出样例 输入样例#1: 7 7 1 7 1 2 2 2 4 2 4 6 2 6 7 2 1 3 2 3 5 4 5 7 2 输出样例#1: 6 输入样例#2: 5 5 1 4 1 2 1 1 3 ...
- [BZOJ5109/CodePlus2017]大吉大利,晚上吃鸡!
Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏中,皮皮和毛毛最喜欢做的事情就是堵桥,每每有一个好时机都能收到不少的快递 ...
随机推荐
- 使用maven如何生成源代码的jar包
http://hw1287789687.iteye.com/blog/1943157 Maven build...
- win7下面安装flex和bison
通过Cygwin工具进行安装 下载地址cygwin 安装截图: 搜索flex,选择相应的版本进行安装 搜索bison 然后下一步 安装完成后,将D:\cygwin64\bin路径添加到PATH环境变量 ...
- 对象数组(JSON) 根据某个共同字段 分组
1.在前端页面用js处理 转载https://www.cnblogs.com/rysinal/p/5834446.html 为了第二种方法对比,特意把id 转化为pid var arr = [ {&q ...
- release,debug库互调用,32位,64位程序与库互调用
以下是基于visual studio 2015和cmake的实验 1,debug或release的应用程序都可以调用release的库2,win32和x64的应用和库无法互调用,在VS中链接时会有一堆 ...
- 从 spring-cloud-alibaba-nacos-config 进入 nacos-client
sc 的 bootstrap context 是 main application context 的 parent,需要在 main application context 中使用的 bean 可以 ...
- 阶段1 语言基础+高级_1-3-Java语言高级_04-集合_06 Set集合_3_HashSet集合存储数据的结构
特点就是查询速度快 jdk1.8以后,如果链表的长度超过了8位.就会把转成红黑树,也是为了提高查询的效率
- Java ——流(Stream)、文件(File)和IO
本节重点思维导图 示例:将指定的字符写到文件中 public static void main(String[] args) { try { //1.创建一个空的文件 File file = new ...
- Zuul网关过滤器中转发头信息不起作用
大家应该都知道在做Zuul网关过滤的时候,如果不在网关过滤时转发头信息,经过网关时头信息就会丢失. 但我用 addZuulRequestHeader转发头信息时,却不起作用,在子服务的controll ...
- Smashing Nodejs 读书笔记(二)
了不起的Node.js:将JavaScript进行到底 书名:SMASHING Node.js : JavaScript Everywhere 原作者:(美)劳奇 Rauch.G 译者:赵静 出版日期 ...
- .Net core 3.0 SignalR+Vue 实现简单的即时通讯/聊天IM (无jq依赖)
.Net core 中的SignalR JavaScript客户端已经不需要依赖Jquery了 一.服务端 1.nuget安装 Microsoft.AspNetCore.SignalR 2.在star ...