链接

  • 最短路径\(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 大吉大利,晚上吃鸡!的更多相关文章

  1. GMA Round 1 大吉大利,晚上吃鸡

    传送门 大吉大利,晚上吃鸡 新年走亲访友能干点啥呢,咱开黑吃鸡吧. 这里有32个人,每个人都可能想玩或者不想玩,这样子一共有$2^{32}$种可能.而要开黑当然得4人4人组一队(四人模式),所以说如果 ...

  2. [BZOJ5109]大吉大利,晚上吃鸡!

    [BZOJ5109]大吉大利,晚上吃鸡! 题目大意: 一张\(n(n\le5\times10^4)\)个点\(m(m\le5\times10^4)\)条边的无向图,节点编号为\(1\)到\(n\),边 ...

  3. 【BZOJ5109】[CodePlus 2017]大吉大利,晚上吃鸡! 最短路+拓扑排序+DP

    [BZOJ5109][CodePlus 2017]大吉大利,晚上吃鸡! Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏 ...

  4. bzoj5109: [CodePlus 2017]大吉大利,晚上吃鸡!

    Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏中,皮皮 和毛毛最喜欢做的事情就是堵桥,每每有一个好时机都能收到不少的快 ...

  5. 「CodePlus 2017 11 月赛」大吉大利,晚上吃鸡!(dij+bitset)

    从S出发跑dij,从T出发跑dij,顺便最短路计数. 令$F(x)$为$S$到$T$最短路经过$x$的方案数,显然这个是可以用$S$到$x$的方案数乘$T$到$x$的方案数来得到. 然后第一个条件就变 ...

  6. BZOJ5109 CodePlus 2017大吉大利,晚上吃鸡!(最短路+拓扑排序+bitset)

    首先跑正反两遍dij求由起点/终点到某点的最短路条数,这样条件一就转化为f(S,A)*f(T,A)+f(S,B)*f(T,B)=f(S,T).同时建出最短路DAG,这样图中任何一条S到T的路径都是最短 ...

  7. 「CodePlus 2017 11 月赛」大吉大利,晚上吃鸡!

    n<=50000,m<=50000的图,给s和t,问有多少点对$(a,b)$满足 嗯. 不会. 首先最短路DAG造出来,然后两个条件转述一下:条件一,$N_a$表示从s到t经过a的路径,$ ...

  8. [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 ...

  9. [BZOJ5109/CodePlus2017]大吉大利,晚上吃鸡!

    Description 最近<绝地求生:大逃杀>风靡全球,皮皮和毛毛也迷上了这款游戏,他们经常组队玩这款游戏.在游戏中,皮皮和毛毛最喜欢做的事情就是堵桥,每每有一个好时机都能收到不少的快递 ...

随机推荐

  1. Linux内核调试方法总结之bugreport

    bugreport [用途]Android性能分析工具,bugreport记录了Android启动过程日志,启动后的系统状态,包括进程列表.内存信息.VM信息等 [使用方法] Adb bugrepor ...

  2. ASP 解析json

    第一个方法是使用 JScript : <script language="jscript" runat="server"> Array.protot ...

  3. 阶段1 语言基础+高级_1-3-Java语言高级_08-JDK8新特性_第3节 两种获取Stream流的方式_11_练习:集合元素处理(Stream方式)

  4. 阶段1 语言基础+高级_1-3-Java语言高级_03-常用API第二部分_第5节 StringBuilder类_1_StringBuilder的原理

    字符串不可变.字符串的缓冲区是可以变的 字符串Sting的底层,被final修饰的不可变的数组 a+b+c最终会产生5个字符串

  5. HAWQ技术总结

    HAWQ技术总结: 1. 官网: http://hawq.incubator.apache.org/ 2. 特性 2.1 sql支持完善 ANSI SQL标准,OLAP扩展,标准JDBC/ODBC支持 ...

  6. unity不规则按钮解决方案

    一种是alpha检测 一种是设置collider 参考: https://zhuanlan.zhihu.com/p/34204396 下面给出第二种方案代码 ///按钮多边形点击方案,注意Canvas ...

  7. Ngix 配置与部署(wsgi,uwsgi,uWSGI)

    1. WSGI 是一种协议接口,他是描述web服务器如何与web应用程序(Django ,Flask ) 通讯的规范. 2. uwsgi 与WSGI协议一样,是uWSGI服务器的独占协议,用于定义传输 ...

  8. C#后台保存Cookie

    一般是: Response.Cookies["backurl"].Expires.AddDays(2); 但是,IE浏览器保存Cookie用 Response.Cookies[&q ...

  9. xmake v2.1.9版本发布,增加可视化图形菜单配置

    此版本主要增加xmake f --menu实现用户自定义图形菜单配置,界面风格类似linux的make menuconfig: [图片上传失败-(image-505bc0-1517795319124) ...

  10. 配置OSPF认证

      按照上图拓扑配置路由器的IP 配置完后测试直连网段连通性 搭建OSPF网络 注意是多区域的配置,R2是ABR 连着area0和area1 并且每个路由器的环回接口IP也要加进去  此时密码以明文方 ...