$Mingqi\_H$

NOIp 2017考挂了...gg

重新开始好了。

计划明年2月24号前复习完所有的NOIp知识点(毕竟很不熟练啊),之后到七月底前学习完省选的东西(flag?)。

从现在开始吧。

11.29 NOIp图论(Ⅰ)

坑:Floyd、Dijkstra、最短路计数、Tarjan、二分图、拓扑。

最短路计数:

类似一个标准的SPFA,不同之处在于加粗的两句:

  • 如果第一次到达nxt节点,nxt节点的最短路数量就等于其前驱节点的最短路数量,当前点需要松弛,则当前点继承被松弛节点的最短路条数,更新一波距离,丢到队列里。
  • 否则当到nxt的最短路长度等于到其前驱点的最短路+当前边的权值时,到nxt点的最短路数量应该加上到其前驱节点的最短路数量(再次更新)。

例题:洛谷P1144,P1608,P3953前30%。

#include<queue>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn=1e5+;
struct Edge{int u,v,w;}edge[*maxn];int head[maxn],cnt;
void add(int u,int v,int w){edge[++cnt].u=head[u],edge[cnt].v=v,edge[cnt].w=w,head[u]=cnt;}
int dis[maxn],vis[maxn];
int ans[maxn];
inline void spfa(int s)
{
queue<int>q;int cur,nxt;
memset(dis,0x7f,sizeof(dis)),memset(vis,,sizeof(vis));
dis[s]=,vis[s]=,q.push(s);
ans[s]=;
while(!q.empty())
{
cur=q.front(),vis[cur]=,q.pop();
for(int i=head[cur];i;i=edge[i].u)
{
nxt=edge[i].v;
if(!ans[nxt])ans[nxt]=ans[cur],dis[nxt]=dis[cur]+edge[i].w,q.push(nxt);
else if(dis[nxt]==dis[cur]+edge[i].w)ans[nxt]+=ans[cur],ans[nxt]%=mod;
}
}
}

以上代码似乎是错误的。gg。

P1608

#include<cstdio>
#include<queue>
#include<cstring>
using namespace std;
const int N = ;
int head[N];
struct node{
int v,w,next;
}edge[N*N/];
int n,e,num=,dis[N];bool vis[N];
void add_edge(int x,int y,int z)
{
edge[++num].v=y;edge[num].w=z;edge[num].next=head[x];head[x]=num;
}
int ans1,cnt[N];
int minn=;
void spfa(int x)
{
queue<int>que;
vis[]=;dis[]=;cnt[]=;
que.push();
while(!que.empty())
{
int u=que.front();que.pop();
if(u==n)continue;
for(int i=head[u];i;i=edge[i].next)
{
int v=edge[i].v;
if(dis[u]+edge[i].w<dis[v])
{
if(dis[u]+edge[i].w<dis[v])
{
dis[v]=dis[x]+edge[i].w;
cnt[v]=cnt[u];
}
if(!vis[v])que.push(v),vis[v]=;
}
else if(dis[u]+edge[i].w==dis[v]) cnt[v]+=cnt[u];
}
vis[u]=;cnt[u]=;
}
}
int main()
{
memset(dis,0x3f,sizeof dis);
scanf("%d%d",&n,&e);
int a,b,c;
for(int i=;i<=e;i++)
{
scanf("%d%d%d",&a,&b,&c);
add_edge(a,b,c);
}
spfa();
if(dis[n]==0x3f3f3f3f)
puts("No Answer");
else printf("%d %d\n",dis[n],cnt[n]);
return ;
}

啊...原来是题目有锅。。。代码没什么问题。

P3953前30%:

#include<queue>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn=+;
struct Edge{int u,v,w;} edge[*maxn];
int head[maxn],c;
void add(int u,int v,int w){edge[++c].u=head[u],edge[c].v=v,edge[c].w=w,head[u]=c;}
int n,m,x,y,z,k,p,t;
int dis[maxn],cnt[maxn],vis[maxn];
int cur,v;
void spfa(int s)
{
memset(dis,0x3f,sizeof(dis)),memset(cnt,,sizeof(cnt)),memset(vis,,sizeof(vis));
queue<int>q;
q.push(s),dis[s]=,cnt[s]=;
while(!q.empty())
{
cur=q.front(),q.pop(),vis[cur]=;
if(cur==n)continue;
for(int i=head[cur]; i; i=edge[i].u)
{
v=edge[i].v;
if(dis[cur]+edge[i].w==dis[v])cnt[v]=(cnt[v]+cnt[i])%p;
if(dis[cur]+edge[i].w<dis[v])dis[v]=dis[cur]+edge[i].w,cnt[v]=cnt[cur];
if(cnt[v]&&!vis[v])vis[v]=,q.push(v);
}
cnt[cur]=;
}
}
int main()
{
scanf("%d",&t);
while(t--)
{
memset(head,,sizeof(head)),c=,memset(edge,,sizeof(edge));
scanf("%d%d%d%d",&n,&m,&k,&p);
while(m--)scanf("%d%d%d",&x,&y,&z),add(x,y,z);
spfa();
printf("%d\n",cnt[n]);
}
return ;
}

由此我们可以得到一般的最短路计数的模板。

void spfa(int s)
{
memset(dis,0x3f,sizeof(dis)),memset(cnt,,sizeof(cnt)),memset(vis,,sizeof(vis));
queue<int>q;
q.push(s),dis[s]=,cnt[s]=;
while(!q.empty())
{
cur=q.front(),q.pop(),vis[cur]=;
if(cur==n)continue;
for(int i=head[cur]; i; i=edge[i].u)
{
v=edge[i].v;
if(dis[cur]+edge[i].w==dis[v])cnt[v]+=cnt[i];
if(dis[cur]+edge[i].w<dis[v])dis[v]=dis[cur]+edge[i].w,cnt[v]=cnt[cur];
if(cnt[v]&&!vis[v])vis[v]=,q.push(v);
}
cnt[cur]=;
}
}

就是一个普通SPFA加了点东西。

NOIp知识点复习——最短路计数的更多相关文章

  1. 2018.11.05 NOIP模拟 规避(最短路计数)

    传送门 正难则反. 考虑计算两人相遇的方案数. 先正反跑一遍最短路计数. 然后对于一条在最短路上的边(u,v)(u,v)(u,v),如果(dis(s,u)*2<total&&di ...

  2. Noip知识点备考

    作为一个oier,适当的整理是有必要的.蒟蒻根据自己的理解,筛选出考noip应当掌握的知识点.可能后期还有解题思路和模板,先挖个坑慢慢补呗. 60级张炳琪Noip知识点总结 可能是本人比较弱,写的内容 ...

  3. NOIP系列复习及题目集合

    首先是我的酱油记了啦~: Xs的NOIP2014酱油记,持续更新中 知识点方面: noip知识点总结之--贪心 noip知识点总结之--线性筛法及其拓展 noip知识点总结之--欧几里得算法和扩展欧几 ...

  4. 【SPFA】 最短路计数

    最短路计数 [问题描述]   给出一个N个顶点M条边的无向无权图,顶点编号为1-N.问从顶点1开始,到其他每个点的最短路有几条. [输入格式]   输入第一行包含2个正整数N,M,为图的顶点数与边数. ...

  5. noip知识点总结之--欧几里得算法和扩展欧几里得算法

    一.欧几里得算法 名字非常高大上的不一定难,比如欧几里得算法...其实就是求两个正整数a, b的最大公约数(即gcd),亦称辗转相除法 需要先知道一个定理: gcd(a, b) = gcd(b, a  ...

  6. P1144 最短路计数

    P1144 最短路计数 题目描述 给出一个N个顶点M条边的无向无权图,顶点编号为1-N.问从顶点1开始,到其他每个点的最短路有几条. 输入输出格式 输入格式: 输入第一行包含2个正整数N,M,为图的顶 ...

  7. 洛谷P1144最短路计数题解

    最短路计数 此题还是寻找从1到i点总共有几个最短路且每条边的边长为1,对于这种寻找最短路的个数,我们可以反向搜索,即先用\(SPFA\)预处理出所有点的最短路,然后我们反向记忆化搜索,可以用\(sum ...

  8. 洛谷P1144 最短路计数(SPFA)

    To 洛谷.1144 最短路计数 题目描述 给出一个N个顶点M条边的无向无权图,顶点编号为1-N.问从顶点1开始,到其他每个点的最短路有几条. 输入输出格式 输入格式: 输入第一行包含2个正整数N,M ...

  9. 洛谷 P1144 最短路计数 解题报告

    P1144 最短路计数 题目描述 给出一个\(N\)个顶点\(M\)条边的无向无权图,顶点编号为\(1-N\).问从顶点1开始,到其他每个点的最短路有几条. 输入输出格式 输入格式: 第一行包含2个正 ...

随机推荐

  1. jstl自己定义函数的使用

    因为本人之前并没有接触过jstl标签,说来也可笑,之前一直使用struts2标签.近期项目用到jstl,所以做些记录方便以后自己查看. jstl的强大原因之中的一个我觉得就是他的自己定义函数,我们能够 ...

  2. IE浏览器 多版本之间切换

    由于Win7系统最低支持IE8的版本,不能通过安装IE7版本来达到要求. 有一个替代方案,通过IE自带Emulation来实现,实现步骤如下: 打开现有的IE浏览器 按下F12键 切换到“Emulat ...

  3. git分支的合并和冲突解决【转】

    本文转载自:http://blog.csdn.net/Kingson_Wu/article/details/39227611 http://gitbook.liuhui998.com/3_3.html ...

  4. uboot流程分析--修改android启动模式按键【转】

    本文转载自:http://blog.csdn.net/dkleikesa/article/details/9792747 本人用的android平台用的bootloader用的是uboot,貌似大多数 ...

  5. algorithm库———count&&countif

    algorithm头文件定义了一个count的函数,其功能类似于find.这个函数使用一对迭代器和一个值做参数,返回这个值出现次数的统计结果. 编写程序读取一系列int型数据,并将它们存储到vecto ...

  6. poj2262 Goldbach's Conjecture——筛素数

    题目:http://poj.org/problem?id=2262 水水更健康~ 代码如下: #include<iostream> #include<cstdio> #incl ...

  7. zabbix监控kafka消费

    一.Kafka监控的几个指标 1.lag:多少消息没有消费 lag=logsize-offset 2.logsize:Kafka存的消息总数 3.offset:已经消费的消息 Kafka管理工具 介绍 ...

  8. 【WIP】Bootstrap nav

    创建: 2017/09/28   更新: 2017/10/14 标题加上[WIP]

  9. bzoj题目分类

    转载于http://blog.csdn.net/creationaugust/article/details/513876231000:A+B 1001:平面图最小割,转对偶图最短路 1002:矩阵树 ...

  10. Sum It Up -- 深搜 ---较难

    每一行都是一组测试案例   第一个数字 表示总和 第二个数字表示 一共有几个可用数据  现在 按照从小到大的顺序   输出  那些数字中若干数字之和为总和的  信息 /. 很好很明显的  遍历痕迹 , ...