ZOJ 3408 Gao
ZOJ题目页面传送门
给定一个有向图\(G=(V,E),|V|=n,|E|=m\)(可能有重边和自环,节点从\(0\)开始编号),以及\(q\)组询问,对于每组询问你需要回答有多少条从节点\(0\)开始的最短路经过节点\(x\)(节点\(0\)到某一个节点的最短路可能不唯一),输出答案的后\(10\)位。本题多测。
\(q,n\in\left[1,10^4\right],m\in\left[1,5\times10^4\right]\),边权为正。
既然题目要数最短路的个数,我们可以把那些不在最短路上的边去掉,只保留在最短路上的边,构造出一个新图\(G'=(V,E')\)。这样问题就转化成了在\(G'\)上有多少条从节点\(0\)开始的路径经过节点\(x\)(显然的吧)。那怎么知道那些边该留那些边不该留呢?我们可以先跑一遍堆优化Dijkstra求出到节点\(0\)的最短路长度数组\(dis\),那么\(E'=\{(x,y,len)\mid(x,y,len)\in E,dis_y=dis_x+len\}\)。
接下来我们就要求在\(G'\)上有多少条从节点\(0\)开始的路径经过节点\(x\)了。我们考虑将一条从节点\(0\)开始、经过节点\(x\)、终点为\(y\)的路径拆成两段:\(0\to x,x\to y\),对它们分别计数,最后相乘即可。很显然,\(G'\)是无环的,也就是一个DAG(因为边权为正),这样就可以在\(G'\)上DP了。设\(dp1_i\)表示路径\(0\to i\)的个数,\(dp2_i\)表示以\(i\)为起点的路径的个数(即\(\sum\limits_{j\in V}\text{路径}i\to j\text{的个数}\))。状态转移方程显然是:
dp1_i&=\begin{cases}1&i=0\\\sum\limits_{(j,i,len)\in E'}dp1_j&i>0\end{cases}\\
dp2_i&=\sum_{(i,j,len)\in E'}dp2_j+1
\end{aligned}
\]
可这是一个图啊,DP顺序是什么呢?容易发现,要想知道\(dp1_i\),得先知道\(\forall j\left((j,i,len)\in E'\right),dp1_j\),所以要按照拓扑序DP;\(dp2\)反之,要按照拓扑逆序DP。(如果你懒得写拓扑排序,也可以记忆化搜索)
这题还有个毒瘤的地方,就是要保留后\(10\)位(即\(\bmod 10^{10}\)),在乘法的时候会到\(10^{20}\),爆unsigned long long。这时聪(rui)明(zhi)的读者可能会写高精度(毕竟只有最后一步才是乘法,不会TLE),但有个更巧妙的方法:利用乘法分配律,显然
\]
这样最大是\(10^5\times10^{10}=10^{15}\)级别的,不会爆。
下面贴AC代码:(最近码风有了很大的改变)
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define mp make_pair
#define X first
#define Y second
#define pb push_back
const int mod=10000000000ll,inf=1e18;
const int N=10000;
int n/*点数*/,m/*边数*/,qu/*数据组数*/;
vector<pair<int,int> > nei[N]/*原邻接表*/;
vector<int> dnei[N]/*新邻接表*/,rnei[N]/*新反邻接表*/;
int dis[N]/*到节点0的最短路长度*/;
bool vis[N]/*访问标记*/;
int ideg[N]/*新图中的入度*/;
void dijkstra(){//堆优化Dijkstra
priority_queue<pair<int,int>,vector<pair<int,int> >,greater<pair<int,int> > > q;
q.push(mp(0,0));
for(int i=1;i<n;i++)dis[i]=inf;
for(int i=0;i<n;i++)vis[i]=false;
while(q.size()){
int x=q.top().Y;
q.pop();
if(vis[x])continue;
vis[x]=true;
for(int i=0;i<nei[x].size();i++){
int y=nei[x][i].X,len=nei[x][i].Y;
if(dis[x]+len<dis[y])q.push(mp(dis[y]=dis[x]+len,y));
}
}
//建新图
for(int i=0;i<n;i++)dnei[i].clear(),rnei[i].clear();
for(int i=0;i<n;i++)for(int j=0;j<nei[i].size();j++){
int x=nei[i][j].X,len=nei[i][j].Y;
if(dis[x]==dis[i]+len)dnei[i].pb(x),rnei[x].pb(i),ideg[x]++;
}
}
vector<int> topo;//拓扑序
void toposort(){//拓扑排序
queue<int> q;
for(int i=0;i<n;i++)if(!ideg[i])q.push(i);
topo.clear();
while(q.size()){
int x=q.front();
q.pop();
topo.pb(x);
for(int i=0;i<dnei[x].size();i++){
int y=dnei[x][i];
ideg[y]--;
if(!ideg[y])q.push(y);
}
}
}
int dp1[N]/*dp1[i]表示新图中路径0->i的个数*/,dp2[N]/*dp2[i]表示新图中路径i->j的个数之和*/;
int prod(int x,int y){//mod 10^5意义下的乘法
int lx=x%100000,ly=y%100000;
return (lx*y+(x-lx)*ly)%mod;
}
void prt(int x){//输出后10位
vector<int> v;
for(int i=1;i<=10;i++)v.pb(x%10),x/=10;
for(int i=9;~i;i--)printf("%lld",v[i]);
}
void mian(){
for(int i=0;i<n;i++)nei[i].clear();
while(m--){
int x,y,z;
scanf("%lld%lld%lld",&x,&y,&z);
nei[x].pb(mp(y,z));
}
dijkstra();
toposort();
for(int i=0;i<n;i++){//dp1,拓扑序
int x=topo[i];
dp1[x]=!x;
for(int j=0;j<rnei[x].size();j++)
(dp1[x]+=dp1[rnei[x][j]])%=mod;
}
for(int i=n-1;~i;i--){//dp2,拓扑逆序
int x=topo[i];
dp2[x]=1;
for(int j=0;j<dnei[x].size();j++)
(dp2[x]+=dp2[dnei[x][j]])%=mod;
}
while(qu--){
int x;
scanf("%lld",&x);
prt(prod(dp1[x],dp2[x]));/*相乘得到总数*/puts("");
}
}
signed main(){
// freopen("C:\\Users\\chenx\\OneDrive\\桌面\\txt.txt","r",stdin);
while(~scanf("%lld%lld%lld",&n,&m,&qu))mian();
return 0;
}
我做这题的时候打错了一个字母,把prod(int,int)函数里的ly=y%100000打成了ly=x%100000,导致我调了几个小时。。。哎,血的教训!
ZOJ 3408 Gao的更多相关文章
- zoj 3672 Gao The Sequence
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=4915题意:a[k]-一个任意的数,这个数要等于a[1]~a[k]每个数减去任意 ...
- ZOJ 3647 Gao the Grid dp,思路,格中取同一行的三点,经典 难度:3
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=4837 三角形的总数=格子中任取3个点的组合数-同一横行任取3个点数目-同一纵行 ...
- zoj 3647 Gao the Grid
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=4837 先求出从所有点随机找出三个点的组合数,然后去掉共线的,平行好去掉,斜线就 ...
- Zoj 3535 Gao the String II (AC自己主动机+dp)
题目大意: 用集合A中的串构造出一个串,使之让很多其它的setB中的串成为他的子串. 思路分析: 和 Codeforces 86C 几乎相同. 只是这里是要用A中的构造. 先用A 和 B的串构造一个自 ...
- ZOJ Monthly, November 2012
A.ZOJ 3666 Alice and Bob 组合博弈,SG函数应用 #include<vector> #include<cstdio> #include<cstri ...
- AC自动机-算法详解
What's Aho-Corasick automaton? 一种多模式串匹配算法,该算法在1975年产生于贝尔实验室,是著名的多模式匹配算法之一. 简单的说,KMP用来在一篇文章中匹配一个模式串:但 ...
- zoj Gao The Sequence
Gao The Sequence Time Limit: 2 Seconds Memory Limit: 65536 KB You are given a sequence of integ ...
- Zoj 3781(构造)
Zoj 3781(构造) Zoj 3781 As we all know, Coach Gao is a talented chef, because he is able to cook M dis ...
- ZOJ People Counting
第十三届浙江省大学生程序设计竞赛 I 题, 一道模拟题. ZOJ 3944http://www.icpc.moe/onlinejudge/showProblem.do?problemCode=394 ...
随机推荐
- Azure中国CDN全球覆盖功能初探
在不久前的4月初,Azure中国官网上简短地发布了其CDN中“标准版 Zone 2”功能.一开始笔者尚有些摸不着头脑,这个“Zone 2”具体指的是什么呢?好在后来官网更新了信息描述如下: 这下就比较 ...
- c++学习书籍推荐《C++ GUI Qt 4编程(第2版)》下载
下载地址:点我 百度云及其他网盘下载地址:点我 编辑推荐 <C++ GUI Qt 4编程(第2版)>讲授的大量Qt4编程原理和实践,都可以轻易将其应用于Qt4.4.Qt4.5及后续版本的Q ...
- 作者联系方式D1
欢迎大伙投稿,审核通过免费发布. 奥特曼超人 KARL-Dujinyang QQ: 309933706 QQ: 1875125470 工作时间都会在线. 偶尔博客,不过有问题可以加Q来找我讨论 ...
- 使用wincc vbs脚本查找进程及如何运行进程
使用vbs代码查看某个进程是否在运行,本文要检查的进程名为 QRscan.exe,其代码如下: sub CheckProcess Dim WMI,Objs,Process,ObjSet WMI=Get ...
- Go语言解密上篇中用java aes实现的加密
上一篇java aes文件加解密中加密的梅须逊雪三分白,雪却输梅一段香.使用go语言解密. 解密代码如下: AESUtil.go package util import ( "crypto/ ...
- VUE+element tree 实现权限管理
先写个标题~ ~,后续有空在写 具体功能: 1. 获取所有角色权限列表展示,点击进行编辑,编辑用terr树形结构显示页面结构 2.提交的数据格式(页面名称,角色ID,父节点ID,子节点ID) 3.后面 ...
- 【题解】跳房子-C++
Description奶牛按不太传统的方式玩起小朋友玩的跳房子游戏,现给出一个5*%的由数字组成的网格.它们在格子中向前前跳,向后跳,向左跳,向右跳,跳到网格中另一个数字后,又这样继续跳(可能跳到某个 ...
- 手机web app开发笔记
各位朋友好,最近自学开发了一个手机Web APP,“编程之路”,主要功能包括文章的展示,留言,注册登录,音乐播放等.为了记录学习心得,提高自己的编程水平,也许对其他朋友有点启发,特整理开发笔记如下. ...
- c++语言常用转义序列符号
\a 响铃 \n 换行符 \r 回车符 t 水平制表符(Tab键) \b 退格符(BackSpace键) \\ 反斜线 \' ...
- Linux vim环境设置
//vim /etc/vimrc(管理员权限) 1. 显示行号: set number 或者 set nu 不显示行号: set nonu 2.自动缩进: set autoindent 3.C语言自 ...