惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

GbyAI
GbyAI
T
Tenable Blog
Webroot Blog
Webroot Blog
L
Lohrmann on Cybersecurity
S
Securelist
S
Schneier on Security
NISL@THU
NISL@THU
Know Your Adversary
Know Your Adversary
C
Cybersecurity and Infrastructure Security Agency CISA
T
The Exploit Database - CXSecurity.com
L
LINUX DO - 热门话题
C
CXSECURITY Database RSS Feed - CXSecurity.com
O
OpenAI News
I
Intezer
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
TaoSecurity Blog
TaoSecurity Blog
S
Secure Thoughts
Application and Cybersecurity Blog
Application and Cybersecurity Blog
P
Privacy International News Feed
H
Hacker News: Front Page
N
Netflix TechBlog - Medium
M
MIT News - Artificial intelligence
博客园 - Franky
PCI Perspectives
PCI Perspectives
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Microsoft Azure Blog
Microsoft Azure Blog
MongoDB | Blog
MongoDB | Blog
L
LangChain Blog
P
Proofpoint News Feed
S
Security Affairs
WordPress大学
WordPress大学
The Last Watchdog
The Last Watchdog
S
SegmentFault 最新的问题
小众软件
小众软件
F
Full Disclosure
博客园 - 叶小钗
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
T
The Blog of Author Tim Ferriss
Simon Willison's Weblog
Simon Willison's Weblog
P
Palo Alto Networks Blog
Security Latest
Security Latest
P
Proofpoint News Feed
月光博客
月光博客
T
Tailwind CSS Blog
Scott Helme
Scott Helme
Hacker News - Newest:
Hacker News - Newest: "LLM"
Google Online Security Blog
Google Online Security Blog
T
Threat Research - Cisco Blogs
Help Net Security
Help Net Security
Project Zero
Project Zero

博客园_首页

Plist 二进制格式 Milvus 和 PGVector,哪个更好? OpenClaw 已过时?在 VS Code 中运行 Hermes Agent! 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent UIImageView 设置图片不生效的原因排查 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里借助MU实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 前端小白一句话,AI 帮我做了个颜值拉满的桌面媒体播放器。当代码不再是门槛,一句话编程就是现实。 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 Claude code入门 - 陈彦斌 一文学习入门 ThingsBoard 开源物联网平台 GitHub 热门项目 | 2026年04月16日 如何为GIT设置全局勾子,为每次提交追加信息 Number.isFinite和isFinite与isNaN()和Number.isNaN的区别 PortSwigger SQL注入LAB2 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) GLM模型这么火,咱们用vllm也咧一个呗! 深入理解 AbortController:从底层原理到跨语言设计哲学 字符串学习笔记 多租户系统框架的基础模块设计和分析设计 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 ElasticSearch7.X版本配置密码 用Manim实现动态交点计算--从一个动点问题说起 团结引擎+Addressable+Instant Game打包抖音小游戏 function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 bubseek —— 让 Agent 的足迹,变成团队的洞察 通过 C# 读取并导出 PDF 书签 如何用 GitHub Actions 实现 Steam 自动化发布 【从0到1构建一个ClaudeAgent】并发-后台任务 .NET 高级开发 | 定制 ASP.NET Core 框架 电子小白:什么是运算放大器(运放) zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 工作的意义、工作的困难认知再思考 .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 开了 TUN 模式还是直连?90% 的人都踩过这个坑 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术
2、BellMan-Ford算法
Carey_ccl · 2026-05-19 · via 博客园_首页

一、BellMan-Ford算法简介

  与Dijkstra算法一样,BellMan-Ford算法也是用于求有向图和无向图的单源最短路径的算法。但是,BellMan-Ford算法与Dijkstra算法的不同处,有以下2点:
①、BellMan-Ford算法可以用于边的权值为负数的有向图中,但该图中不能存在负值圈,也就是说,BellMan-Ford算法在计算无向图时,该无向图不能存在负值边(无向图的负值边就会存在负值圈),Dijkstra算法所计算的有向图和无向图中,边的权值都不能为负数;
②、BellMan-Ford算法是通过对边进行|V|-1次松弛操作来实现的,|V|是图中结点的数量,而Dijkstra算法是通过贪心法,选取未处理的结点中权值最小的结点,然后对选取的结点的邻接边进行松弛操作。

1.1、BellMan-Ford算法的松弛函数

  对于边集合E中的任意边,以w(u,v)表示结点u出发到结点v的边(Edge)的权值,以d[v]表示当前从起点s到结点v的路径权值,若存在边w(u,v),使得:

\[d[v]>d[u]+w(u,v) \]

则更新d[v]的值:

\[d[v]=d[u]+w(u,v) \]

所以松弛函数的作用,就是判断是否经过某个顶点,或者说经过某条边,可以缩短起点到终点的路径权值。

1.2、BellMan-Ford算法中松弛函数的执行次数

  现有所有结点a~d,如下图所示:
image

  用d[d]表示起点a到结点d的距离,用δ(a,d)表示起点a到结点d的最短路径权值,用p=<a,...,d> 表示结点a到结点d的路径,初始情况d[a]=0,d[v]=∞,v∈V-{a}。以对边集合 E 中每条边执行一次松弛函数作为一次迭代。那么,最好情况和最坏情况的分析,分别如下:
①、最好情况下,如果遍历松弛边的顺序为:w(a,b),w(b,c),w(c,d),其他两条边w(a,c),w(a,d)顺序无影响;

  • 第一次迭代
对边w(a,b)执行松弛函数,则d[b] = d[a]+w(a,b) = 1;
对边w(b,c)执行松弛函数,则d[c] = d[b]+w(b,c) = 3;
对边w(c,d)执行松弛函数,则d[d] = d[c]+w(c,d) = 8;

因为图结构比较简单,所以可以直接由观察得知,经过第一次迭代,即可得出从起点a到所有结点b、c、d的最短路径权值。
②、最坏情况下,如果遍历松弛边的顺序为:w(c,d),w(b,c)或w(b,c),w(c,d),其他三条边w(a,b),w(a,c),w(a,d)顺序无影响;

  • 第一次迭代
对边w(c,d)执行松弛函数,则d[d] = ∞;
对边w(b,c)执行松弛函数,则d[c] = ∞;
对边w(a,b)执行松弛函数,则d[b] = d[a] + w(a,b) = 1;
对边w(a,c)执行松弛函数,则d[c] = d[a] + w(a,c) = 6;
对边w(a,d)执行松弛函数,则d[d] = d[a] + w(a,d) = 10; 

第一次迭代,有三条边起到了松弛的效果,直观的可以看出 d[b] = δ(a,b),第一次迭代可以获得起点a到结点b的最短路径,路径为 P=<a,b>;

  • 第二次迭代
对边w(c,d)执行松弛函数,则d[d] = 10;
对边w(b,c)执行松弛函数,则d[c] = d[b]+w(b,c) = 3;
对边w(a,b)执行松弛函数,则d[b] = 1;
对边w(a,c)执行松弛函数,则d[c] = 6;
对边w(a,d)执行松弛函数,则d[d] = 10;

第二次迭代,有一条边起到了松弛的效果,直观的可以看出 d[c] = δ(a,c),第二次迭代可以获得起点a到结点b、结点c的最短路径,路径为 P=<a,b,c>;

  • 第三次迭代
对边w(c,d)执行松弛函数,则d[d] = d[c]+w(c,d) = 8;
对边w(b,c)执行松弛函数,则d[c] = 3;
对边w(a,b)执行松弛函数,则d[b] = 1;
对边w(a,c)执行松弛函数,则d[c] = 6;
对边w(a,d)执行松弛函数,则d[d] = 10;

第三次迭代,有一条边起到了松弛的效果,直观的可以看出 d[d] = δ(a,d),第三次迭代可以获得起点a到结点b、结点c、结点d的最短路径,路径为 P=<a,b,c,d>;

1.3、迭代次数分析(反证法)

  详细过程,请查看:https://www.jianshu.com/p/b876fe9b2338

二、代码实现

  有 n 个城市通过一些航班连接。给你一个数组 flights ,其中 flights[i] = [fromi, toi, pricei] ,表示该航班都从城市 fromi 开始,以价格 pricei 抵达 toi。现在给定所有的城市和航班,以及出发城市 src 和目的地 dst,你的任务是找到出一条最多经过 k 站中转的路线,使得从 src 到 dst 的 价格最便宜 ,并返回该价格。 如果不存在这样的路线,则输出 -1。
image
image
image

2.1、方法一

  Bellman Ford + 类(模拟边)的代码实现,如下所示:

class Solution {
    private class Edge {
        int src;
        int dest;
        int cost;

        public Edge(int src, int dest, int cost) {
            this.src = src;
            this.dest = dest;
            this.cost = cost;
        }
    }

    int N = 110;
    int INF = 0x3f3f3f3f;
    int limit, src, dest;
    ArrayList<Edge> list = null;
    int[] distance = new int[N];

    public int findCheapestPrice(int n, int[][] flights, int src, int dst, int k) {
        this.limit = k + 1;
        this.src = src;
        this.dest = dst;
        list = new ArrayList<Edge>();
        for (int[] flight : flights) {
            Edge edge = new Edge(flight[0], flight[1], flight[2]);
            list.add(edge);
        }
        int ans = dp();
        return ans > INF/2 ? -1 : ans;
    }

    public int dp() {
        Arrays.fill(distance, INF);
        distance[src] = 0;
        for (int i = 0; i < limit; i++) {
            int[] clone = distance.clone();
            for (Edge edge : list) {
                int src = edge.src, dest = edge.dest, cost = edge.cost;
                distance[dest] = Math.min(distance[dest], clone[src] + cost);
            }
        }
        return distance[dest];
    }
}
时间复杂度 共进行 k + 1次迭代,每次迭代备份数组复杂度为 O(n),n为所有结点的数量,然后遍历所有的边进行松弛操作,复杂度为 O(m),m为所有边的数量。整体复杂度为 O(k * (n + m))
空间复杂度 O(n + m),n为所有结点的数量,m为所有边的数量
2.2、方法二

  直接利用数组 flights 的Bellman Ford算法,如下所示:

class Solution {
    public int findCheapestPrice(int n, int[][] flights, int src, int dst, int k) {
        int INF = 0x3f3f3f3f;
        int limit = k + 1;
        int[] distance = new int[n];
        Arrays.fill(distance, INF);
        distance[src] = 0;
        for (int i = 0; i < limit; i++) {
            int[] clone = distance.clone();
            for (int[] flight : flights) {
                int s = flight[0], d = flight[1], cost = flight[2];
                distance[d] = Math.min(distance[d], clone[s] + cost);
            }
        }
        return distance[dst] > INF / 2 ? -1 : distance[dst];
    }
}
时间复杂度 共进行 k + 1次迭代,每次迭代备份数组复杂度为 O(n),n为所有结点的数量,然后遍历所有的边进行松弛操作,复杂度为 O(m),m为所有边的数量。整体复杂度为 O(k * (n + m))
空间复杂度 O(n + m),n为所有结点的数量,m为所有边的数量