【牛站 / USACO2007】

news/2025/1/15 14:53:07/

题目


在这里插入图片描述



思路


离散化(降低空间复杂度)

点的编号 ∈ [ 1 , 1000 ] ,但是点的个数最多为 2 ⋅ T ∈ [ 4 , 200 ] 点的编号 \in [1, 1000],但是点的个数最多为 2 \cdot T \in[4, 200] 点的编号[1,1000],但是点的个数最多为2T[4,200]
因此采用离散化处理可以降低空间复杂度 因此采用离散化处理可以降低空间复杂度 因此采用离散化处理可以降低空间复杂度
采用映射unordered_map + 计数器n


类Floyd算法算法核心)

首先回顾一下Floyd算法

for(int k = 1; k <= n; k++)
{for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){g[i][j] = min(g[i][j], g[i][k] + g[k][j]);}}
}

算法的思路是考虑各个中转点来不断更新最短路径


可能会有疑问是关于中转点的遍历时机
但是只要记住一句话,路径可以按照任意结合的顺序形成:以 i 为中转点的大路径可能不会在 k = i 就被更新好,但是他所需要的,中转点为 i 的部分一定会在 k = i 之后被更新好



类Floyd

for(int k = 1; k <= n; k++){for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){temp[i][j] = min(temp[i][j], a[i][k] + b[k][j]);}}}

算法的思路是考虑各个中转点来更新最短路径


两者的区别在于,类Floyd不会拿用前一个中转点更新的路径继续更新以其他点为中转点的路径,换句话说:类Floyd只能用两个旧的状态迭代新的状态,不能使用新的状态进行迭代
这有一个好处:如果 g g g 的初始状态集合是 k k k 条边路径的集合,那么进行一次迭代(类Floyd)之后,新状态集合为 k + k k+k k+k 条边路径的集合,由此,我们就可以控制一条路径含边的数量了



快速幂(降低时间复杂度)

k条边的路径可以按照任意结合的顺序形成
例如: ( a ⇒ b ) ⇒ ( b ⇒ c ) ⇒ ( c ⇒ d ) ( 1 + 1 + 1 ),也可以( a ⇒ b ) ⇒ ( b ⇒ c ⇒ d ) ( 1 + 2 ),更可以 ( a ⇒ b ⇒ c ) ⇒ ( c ⇒ d ) ( 2 + 1 ) 例如: (a \Rightarrow b) \Rightarrow (b \Rightarrow c) \Rightarrow (c \Rightarrow d) (1+1+1),也可以 (a \Rightarrow b) \Rightarrow \; ( b \Rightarrow c \Rightarrow d) (1+2), 更可以 (a \Rightarrow b \Rightarrow c) \Rightarrow (c\Rightarrow d) (2+1) 例如:(ab)(bc)(cd)1+1+1),也可以(ab(bcd)1+2),更可以(abc)(cd)2+1


因此我们将k作二进制唯一分解。将g对应的边长值不断翻番,并且每次将g的边合并到ans中......
最终我们就可以得到对应边长值为k的邻接矩阵ans



关于一些细节
if(!id.count(S)) id[S] = ++n;S = id[S]; //为什么不写到if里?if(!id.count(E)) id[E] = ++n;E = id[E];

哪怕之前进行过离散化处理,此时点的编号也要更新才对啊。不能够说离散化处理了这个 id 就好了,我不用离散化之后的S,我还用原来的S


for (int i = 1; i <= n; i ++ ) ans[i][i] = 0; //为啥g[i][i]不能是0,这个非得是0

ans[i][i]必须为0是因为要mul(ans, ans, g),即将ans 和 g 中的路径作合并,只有ans[i][i] = 0, ans[i][j] = ans[i][i] + g[i][j]才能够生效,将g[i][j]保存在ans中,否则 g 中 的一些路径没有按照预期进入ans中


同理,g 中不搞 g[i][i] = 0 是因为 g 的初态表示边为1的路径,i 到 i 算边为 0。如果搞g[i][i] = 0,mul(g, g, g) 迭代一次后,会存在边为(0+1)的路径,我们就丧失了对于边这个参数的掌控



代码


#include <bits/stdc++.h>
using namespace std;const int N = 210;int g[N][N], ans[N][N];
unordered_map<int, int> id;
int k, m, S, E, n;void mul(int c[][N], int a[][N], int b[][N])
{int temp[N][N];memset(temp, 0x3f, sizeof temp);for(int k = 1; k <= n; k++){for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){temp[i][j] = min(temp[i][j], a[i][k] + b[k][j]);}}}memcpy(c, temp, sizeof temp);
}
void qmi()
{memset(ans, 0x3f, sizeof ans);for (int i = 1; i <= n; i ++ ) ans[i][i] = 0;while(k){if(k & 1) mul(ans, ans, g);mul(g, g, g);k >>= 1;}
}
int main()
{cin >> k >> m >> S >> E;if(!id.count(S)) id[S] = ++n;S = id[S];if(!id.count(E)) id[E] = ++n;E = id[E];memset(g, 0x3f, sizeof g);for(int i = 1; i <= m; i++){int a, b, c;cin >> c >> a >> b;if(!id.count(a)) id[a] = ++n;a = id[a];if(!id.count(b)) id[b] = ++n;b = id[b];g[a][b] = g[b][a] = min(g[a][b], c);}qmi();cout << ans[S][E];return 0;
}

http://www.ppmy.cn/news/1522270.html

相关文章

手机扬声器音量总是不够大?试试“扬声器助推器”吧

手机的扬声器音量总是不够大&#xff0c;尤其是在嘈杂的环境中&#xff0c;音乐和视频的声音总是不太清晰。直到我发现了这款“扬声器助推器”&#xff0c;我的手机音质瞬间提升了好几个档次。 软件简介&#xff1a; “扬声器助推器”利用先进的音频处理技术&#xff0c;能够…

扑捉一只耿鬼(HTML文件)

图例&#xff1a; 代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><title>耿鬼</title><style>body {background: #fff;font-family: Comfortaa, sans-serif;}* {box-sizing:…

微信小程序请求数据接口封装

文章目录 前言一、方法参考站二、使用步骤1.首先需要创建api文件夹&#xff0c;在文件夹里创建api.js文件2.修改app.js3.页面里使用 总结 前言 最近在写小程序项目&#xff0c;为了节约代码量&#xff0c;以及为了防止后期多处修改地址容易出问题或者遗漏&#xff0c;所以对数…

Maven:简化Java项目管理的利器

Maven&#xff1a;简化Java项目管理的利器 在现代Java开发中&#xff0c;项目管理和构建工具扮演着至关重要的角色。其中&#xff0c;Maven无疑是最受欢迎和广泛使用的工具之一。本文将深入探讨Maven的核心概念、配置方法以及在实际开发中的应用&#xff0c;帮助您更好地理解和…

html css网页制作

​ 大家好&#xff0c;我是程序员小羊&#xff01; 前言&#xff1a; HTML 和 CSS 是制作网页的基础。HTML 用于定义网页的结构和内容&#xff0c;CSS 用于设计网页的样式和布局。以下是一个详细的网页制作成品教程&#xff0c;包括 HTML 和 CSS 的基础知识&#xff0c;及如何…

如何在算家云搭建OpenSora 1.2(文本生成视频)

一. OpenSora 1.2简介 1. 技术特点 高清视频生成 &#xff1a; OpenSora 1.2 在 720p 高清文生视频质量和生成时长上取得了突破性进展&#xff0c;支持无缝产出任意风格的高质量短片。通过引入视频压缩网络&#xff08;VAE&#xff09;和更优的扩散模型算法&#xff0c;显著…

数据传输安全——混合加解密

使用Hutool实现AES与RSA混合加密解密——构建安全的数据传输通道 在当今数字化社会中&#xff0c;信息安全已经成为企业和个人不可忽视的重要议题。加密技术作为保障数据安全的重要手段&#xff0c;其作用愈发突出。本文将深入探讨如何利用Hutool库实现AES与RSA混合加密解密方…

跨平台RTSP播放器之VLC Media Player还是SmartPlayer?

好多开发者纠结&#xff0c;RTSP流播放&#xff0c;到底是用开源的VLC Media Player还是大牛直播SDK的SmartPlayer&#xff1f;针对此&#xff0c;本文做个简单的技术探讨&#xff0c;方便开发者根据实际需要&#xff0c;做适合自己场景的选择&#xff1a; VLC Media Player …