题目链接


Solution

  1. 非正解

    似乎比较蛇啊,先个一个部分分做法,最短路+\(DP\).

    在求最短路的堆或者队列中存储元素 \(dis_{i,j}\) 代表 \(i\) 这个节点,走了 \(j\) 条边的距离.

    然后跑堆优化 \(Dijkstra\) 或者 SPFA 即可.

    复杂度 \(O(N*nlog(n))\).

    其中 \(N\) 代表要走的边条数, \(n\) 代表节点数 . 但是这题 \(N\) 有 \(10^6\) ...

    洛谷上可以过掉 \(49\) 分.

    不过只要 \(N\) 小一点,这个算法明显更优啊...

  2. 正解

    虽然这题 \(N\) 很大,但是 \(n\) 超级小啊... 只有 \(10^2\) ...

    于是可以用 \(Floyd\) 跑最短路,然后呢??

    倍增。

    令 \(dis_{i,j,k}\) 代表从 \(i\) 到 \(j\) 走了 \(2^k\) 条边的最小距离.

    然后 \(Floyd\) 转移的时候利用和倍增跳 LCA 差不多的方式转移即可.

    (具体可以看代码)

    时间复杂度 \(O(n^3*log(N))\).

另外此题还需要一点离散化.


Code

49分:

#include<bits/stdc++.h>
using namespace std;
const int maxn=10008;
struct sj{int to,next,w;}a[maxn];
struct node{int u,k;};
int head[maxn],size,cnt;
int x[maxn],y[maxn],w[maxn];
int to[maxn],num,n,t,S,E;
map<int,int>dis[508],v[508];
map<int,int>js[508];
map<int,int>kkk; int read()
{
char ch=getchar(); int f=1,w=0;
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch<='9'&&ch>='0'){w=w*10+ch-'0';ch=getchar();}
return f*w;
} void add(int x,int y,int w)
{
a[++size].to=y;
a[size].next=head[x];
head[x]=size;
a[size].w=w;
} void spfa()
{
queue<node>q;
q.push((node){S,0});
dis[S][0]=0; v[S][0]=1;
js[S][0]++;
while(!q.empty())
{
node x=q.front();q.pop();
int u=x.u,k=x.k;
if(k==n)continue;
for(int i=head[u];i;i=a[i].next)
{
int tt=a[i].to;
if(!js[tt][k+1])dis[tt][k+1]=192608173,js[tt][k+1]++;
if(dis[tt][k+1]>dis[u][k]+a[i].w)
{
dis[tt][k+1]=dis[u][k]+a[i].w;
if(!v[tt][k+1])
{
q.push((node){tt,k+1});
v[tt][k+1]=1;
}
}
}
v[u][k]=0;
}
} void pre()
{
sort(to+1,to+num+1);
int last=192608173;
for(int i=1;i<=num;i++)
{
if(to[i]!=last)
{
kkk[to[i]]=++cnt;
last=to[i];
}
}
for(int i=1;i<=t;i++)
{
x[i]=kkk[x[i]],y[i]=kkk[y[i]];
//cout<<x[i]<<' '<<y[i]<<endl;
add(x[i],y[i],w[i]);
add(y[i],x[i],w[i]);
}
} int main()
{
n=read(); t=read(); S=read(); E=read();
for(int i=1;i<=t;i++)
{
w[i]=read();
x[i]=read(); y[i]=read();
to[++num]=x[i]; to[++num]=y[i];
}
pre(); S=kkk[S],E=kkk[E];
spfa();
cout<<dis[E][n]<<endl;
}


100分:

#include<bits/stdc++.h>
using namespace std;
const int maxn=10008;
struct node{int u,k;};
int cnt;
int x[maxn],y[maxn],w[maxn];
int to[maxn],num,n,t,S,E;
int dis[508][508][22],ans[508][508];
map<int,int>kkk; int read()
{
char ch=getchar(); int f=1,w=0;
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch<='9'&&ch>='0'){w=w*10+ch-'0';ch=getchar();}
return f*w;
} void floyd()
{
for(int p=1;p<=20;p++)
for(int k=1;k<=cnt;k++)
for(int i=1;i<=cnt;i++)
for(int j=1;j<=cnt;j++)
dis[i][j][p]=min(dis[i][j][p],dis[i][k][p-1]+dis[k][j][p-1]);
} void getans()
{
memset(ans,127/2,sizeof(ans));
int t=0,p=0;
ans[0][S]=0;
while(n)
{
if(n&1)
{
t=!t;
for(int i=1;i<=cnt;i++)
{
ans[t][i]=2e9;
for(int j=1;j<=cnt;j++)
ans[t][i]=min(ans[t][i],ans[!t][j]+dis[i][j][p]);
}
}
p++; n/=2;
}
cout<<ans[t][E];
}
#include<bits/stdc++.h>
using namespace std;
const int maxn=10008;
struct node{int u,k;};
int cnt;
int x[maxn],y[maxn],w[maxn];
int to[maxn],num,n,t,S,E;
int dis[508][508][22],ans[508][508];
map<int,int>kkk; int read()
{
char ch=getchar(); int f=1,w=0;
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch<='9'&&ch>='0'){w=w*10+ch-'0';ch=getchar();}
return f*w;
} void floyd()
{
for(int p=1;p<=20;p++)
for(int k=1;k<=cnt;k++)
for(int i=1;i<=cnt;i++)
for(int j=1;j<=cnt;j++)
dis[i][j][p]=min(dis[i][j][p],dis[i][k][p-1]+dis[k][j][p-1]);
} void getans()
{
memset(ans,127/2,sizeof(ans));
int t=0,p=0;
ans[0][S]=0;
while(n)
{
if(n&1)
{
t=!t;
for(int i=1;i<=cnt;i++)
{
ans[t][i]=2e9;
for(int j=1;j<=cnt;j++)
ans[t][i]=min(ans[t][i],ans[!t][j]+dis[i][j][p]);
}
}
p++; n/=2;
}
cout<<ans[t][E];
} void pre()
{
sort(to+1,to+num+1);
int last=192608173;
for(int i=1;i<=num;i++)
{
if(to[i]!=last)
{
kkk[to[i]]=++cnt;
last=to[i];
}
}
for(int i=1;i<=t;i++)
{
x[i]=kkk[x[i]],y[i]=kkk[y[i]];
dis[x[i]][y[i]][0]=dis[y[i]][x[i]][0]=w[i];
}
} int main()
{
n=read(); t=read(); S=read(); E=read();
memset(dis,127/3,sizeof(dis));
for(int i=1;i<=t;i++)
{
w[i]=read();
x[i]=read(); y[i]=read();
to[++num]=x[i]; to[++num]=y[i];
}
pre(); S=kkk[S],E=kkk[E];
floyd();
getans();
} void pre()
{
sort(to+1,to+num+1);
int last=192608173;
for(int i=1;i<=num;i++)
{
if(to[i]!=last)
{
kkk[to[i]]=++cnt;
last=to[i];
}
}
for(int i=1;i<=t;i++)
{
x[i]=kkk[x[i]],y[i]=kkk[y[i]];
dis[x[i]][y[i]][0]=dis[y[i]][x[i]][0]=w[i];
}
} int main()
{
n=read(); t=read(); S=read(); E=read();
memset(dis,127/3,sizeof(dis));
for(int i=1;i<=t;i++)
{
w[i]=read();
x[i]=read(); y[i]=read();
to[++num]=x[i]; to[++num]=y[i];
}
pre(); S=kkk[S],E=kkk[E];
floyd();
getans();
}

[USACO07NOV]牛继电器Cow Relays (最短路,DP)的更多相关文章

  1. [USACO07NOV]牛继电器Cow Relays

    题目描述 给出一张无向连通图,求S到E经过k条边的最短路. 输入输出样例 输入样例#1: 2 6 6 4 11 4 6 4 4 8 8 4 9 6 6 8 2 6 9 3 8 9 输出样例#1: 10 ...

  2. P2886 [USACO07NOV]牛继电器Cow Relays

    题目描述 For their physical fitness program, N (2 ≤ N ≤ 1,000,000) cows have decided to run a relay race ...

  3. 洛谷P2886 [USACO07NOV]牛继电器Cow Relays

    题意很简单,给一张图,把基本的求起点到终点最短路改成求经过k条边的最短路. 求最短路常用的算法是dijkstra,SPFA,还有floyd. 考虑floyd的过程: c[i][j]=min(c[i][ ...

  4. Luogu 2886 [USACO07NOV]牛继电器Cow Relays

    BZOJ 1706权限题. 倍增$floyd$. 首先这道题有用的点最多只有$200$个,先离散化. 设$f_{p, i, j}$表示经过$2^p$条边从$i$到$j$的最短路,那么有转移$f_{p, ...

  5. [LUOGU] P2886 [USACO07NOV]牛继电器Cow Relays

    https://www.luogu.org/problemnew/show/P2886 给定无向连通图,求经过k条边,s到t的最短路 Floyd形式的矩阵乘法,同样满足结合律,所以可以进行快速幂. 离 ...

  6. [luoguP2886] [USACO07NOV]牛继电器Cow Relays(矩阵)

    传送门 矩阵快速幂,本质是floyd 把 * 改成 + 即可 注意初始化 因为只有100条边,所以可以离散化 #include <cstdio> #include <cstring& ...

  7. 洛谷 P2886 [USACO07NOV]牛继电器Cow Relays

    题面 解题思路 ## floyd+矩阵快速幂,跟GhostCai爷打赌用不用离散化,最后完败..GhostCai真是tql ! 有个巧妙的方法就是将节点重新编号,因为与节点无关. 代码 #includ ...

  8. luogu题解 P2886 【牛继电器Cow Relays】-经过K边最短路&矩阵

    题目链接: https://www.luogu.org/problemnew/show/P2886 Update 6.16 最近看了下<算法导论>,惊奇地发现在在介绍\(APSP\) \( ...

  9. [洛谷P2886] 牛继电器Cow Relays

    问题描述 For their physical fitness program, N (2 ≤ N ≤ 1,000,000) cows have decided to run a relay race ...

随机推荐

  1. db2的定时备份

    定时任务: db2.bat db2cmd -i -w db2_backup.bat exit db2_backup.bat db2 connect to TEST db2 force applicat ...

  2. IDEA设置每次打开重新选择项目

    通过这里,选择settings,或者进入之后的FILE->settings.搜索System 即可出现

  3. VIM 编辑器 -使用详解记录

    1.什么是 vim? Vim是从 vi 发展出来的一个文本编辑器.代码补完.编译及错误跳转等方便编程的功能特别丰富,在程序员中被广泛使用.简单的来说, vi 是老式的字处理器,不过功能已经很齐全了,但 ...

  4. Docker DockerFile文件指令 & 构建

    1.dockerfile指令格式 # Comment注释 INSTRUCTION argument指令名 + 参数 2.普通指令 1. FROM 已存在的镜像,基础镜像,第一条非注释指令 FROM & ...

  5. 02Qt信号与槽(1)

    信号与槽 1.概述 ​ 信号和槽机制是 Qt 的核心机制,信号和槽是一种高级接口,应用于对象之间的通信,它是 Qt 的核心特性,也是 Qt 区别于其他工具包的重要地方.信号和槽是 Qt 自行定义的一种 ...

  6. centos7安装pgsql及操作命令

    1.下载所需要的数据库版本https://yum.postgresql.org/repopackages.php 2.安装数据库版本包 yum install -y https://download. ...

  7. 如何用纯 CSS 创作一个方块旋转动画

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/gjgyWm 可交互视频 ...

  8. php通过geohash算法实现查找附近的商铺

    geohash有以下几个特点: 首先,geohash用一个字符串表示经度和纬度两个坐标.利用geohash,只需在一列上应用索引即可. 其次,geohash表示的并不是一个点,而是一个矩形区域.比如编 ...

  9. 【android】安卓开发apk列表

    - 谷歌的Zxing框架的扫码软件 (目前国内的应用商店很少此种类型的扫码app) - 解析IP地址功能,从IP地址(子网掩码)自动解析出网段,广播地址

  10. Word 借助VBA一键实现插入交叉引用

    最近写论文的时候,经常需要向上或向下插入题注的交叉引用,word 自带的界面往往需要操作多次,才能实现插入.而平时使用较多的只是交叉引用附近的题注,比如如图1.1所示,在图1.1中等,距离较远的引用则 ...