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

推荐订阅源

P
Palo Alto Networks Blog
云风的 BLOG
云风的 BLOG
小众软件
小众软件
V
Visual Studio Blog
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
腾讯CDC
Microsoft Security Blog
Microsoft Security Blog
K
Kaspersky official blog
C
Cisco Blogs
The Last Watchdog
The Last Watchdog
宝玉的分享
宝玉的分享
IT之家
IT之家
Cisco Talos Blog
Cisco Talos Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
W
WeLiveSecurity
NISL@THU
NISL@THU
爱范儿
爱范儿
AI
AI
Security Latest
Security Latest
T
The Blog of Author Tim Ferriss
M
MIT News - Artificial intelligence
博客园 - Franky
B
Blog RSS Feed
GbyAI
GbyAI
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
Engineering at Meta
Engineering at Meta
S
Secure Thoughts
Recorded Future
Recorded Future
L
Lohrmann on Cybersecurity
Webroot Blog
Webroot Blog
C
CERT Recently Published Vulnerability Notes
P
Privacy International News Feed
T
Troy Hunt's Blog
L
LangChain Blog
P
Privacy & Cybersecurity Law Blog
Last Week in AI
Last Week in AI
Know Your Adversary
Know Your Adversary
The Cloudflare Blog
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
www.infosecurity-magazine.com
www.infosecurity-magazine.com
P
Proofpoint News Feed
B
Blog
O
OpenAI News
Latest news
Latest news
T
Tor Project blog
Google DeepMind News
Google DeepMind News
F
Fortinet All Blogs
量子位
博客园 - 三生石上(FineUI控件)
Y
Y Combinator Blog

博客园_首页

Linux实操--组管理、权限管理和定时任务 Java + EasyExcel 实现单个接口导出多个Excel Mem0 源码解析系列(二):提示词工程的深度剖析 Openclaw TaskFlow究竟是什么?和普通Skill技能有什么区别 博文阅读密码验证 - 博客园 嘉立创开源:应该是全网MicroPython教程最多的开发板 Hermes Agent 集成实践:从协议到生产 2026年AI编程工具横评:Cursor、Codex、Claude Code、Zed、Windsurf Java程序员必看的RAG入门教程 2026 AI效率神器:Superpowers + Claude Code 保姆级教程 本地大模型部署全攻略:从 0 到 1 玩转 Ollama 【从0到1构建一个ClaudeAgent】内存管理-上下文压缩 .NET 高级开发 | 设计、实现一个事件总线框架 电子小白入门之NE555 3. WorkBuddy:隐藏玩法,一键召唤专家,让 AI 以"专家身份"给你干活 和AI一起搞事情#3:Claude Teammate 游戏开发翻车实录 【OpenClaw】通过 Nanobot 源码学习架构---(7)Memory C# .NET 周刊|2026年3月3期 我在 Debian 11 上把 K8s 单机搭起来了,过程没你想的那么顺(/opt 目录版) 深度学习进阶(七)Data-efficient Image Transformer CLI+Skill搭建浏览器AI自动化框架,告别一切重复枯燥任务 告别Token账单无底洞:OpenClaw本地部署,重塑企业数据主权的唯一解 FastAPI+Vue:文件分片上传+秒传+断点续传,这坑我帮你踩平了! SBTI 爆火后,我做了个程序员版的 CBTI。。已开源 + 附开发过程 多模态检索开始进入工程期:用 Sentence Transformers 搭建可落地的 Multimodal RAG 100多行代码实现一个最简单的Agent(用ReAct) Claude Code 通关手册(八):推荐 5 个 Hooks,代码质量提升 3 倍 老板:“有人截图了!”。安全部门:“收到,马上查暗水印!” - why技术 技术之外,皆是人间 C#/.NET/.NET Core技术前沿周刊 | 第 69 期(2026年4.01-4.12) Snack JSONPath 项目架构分析 Claude Code Buddy 小析:一个非核心功能,如何体现产品的细节完成度 AI新时代下的图床管理方案-Cloudflare图床+MCP+Skills方案指南 化繁为简:顺丰速运App如何通过 HarmonyOS SDK实现专业级空间测量 从零实现富文本编辑器#13-React非编辑节点的内容渲染 AI开发-python-langchain框架(3-23-OpenAI Functions风格Tool Calling智能助手) .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 PbootCMS 网站内容数量多导致访问慢?这些实用优化方案帮你提速! - 家兴网络技术工作室 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 网站漏洞怎么发现并修复?一篇实用指南(附完整流程) - 家兴网络技术工作室 开了 TUN 模式还是直连?90% 的人都踩过这个坑 Github日报|2026年04月12日 - AI一族 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术 Etsy 把 1000 个 MySQL 分片迁进 Vitess:425TB 数据背后的真正问题不是性能,而是运维规模 MicroPython LVGL基础知识和概念:底层渲染与性能优化 - FreakStudio 数据库草图算法 Python 潮流周刊#146:CPython 引入 Rust 的进展 - 豌豆花下猫 最小生成树 - mofei1116 红日靶场七:从外网入口、容器逃逸到 AD 接管的完整利用链复盘 - YouDiscovered1t 分享四款开源且实用的 Kafka 管理工具 - 追逐时光者 vLLM 权重加载机制全解析:从挑战到理想架构 LCT 学习笔记 - ACehomoxue Avalonia UI 12.0.0 正式发布:架构演进和性能飞跃 - 张善友 当 AI Agent 把调用链拉长,延迟开始成为一门生意 conhost.exe 无法显示 U+2717 - 145a 太秀了,我把自己蒸馏成了 Skill!已开源 - 程序员鱼皮 ASP.NET Core 内存缓存实战:一篇搞懂该怎么配、怎么避坑 基于 Ghostty 带有分割标签页和为 Claude 编程设计的通知终端 - BugShare AI 焊死入口:教育的“操作系统级”重塑 - 郝hai 初级Java开发工程师使用sql脚本编写代码的过程是简单而且不糊涂 - CoderOilStation Claude Code通关手册(六):MCP协议完全指南 - 暮色之狐 边框灯光环绕动画特效实现指南 - Newbe36524 开源:子木蒸馏版的 SEO 审计工具 seo-audit-skill v1.0 我所理解的Python元模型 【从0到1构建一个ClaudeAgent】规划与协调-TodoWrite - 程序员Seven Claude 和 Codex 在审计 Skill 上性能差异探究 - ACai_sec AScript如何实现中文脚本引擎 - rockey627 【渗透测试】HTB Season10 Garfield 全过程wp - dynasty_chenzi Android 开发者为什么必须掌握 AI 能力?端侧视角下的技术变革 树状数组正确性证明 - AC-wyr 你的 AI 焦虑,可能比 AI 本身更危险——ATM 机没有消灭银行柜员,但恐慌消灭了你的判断力 - 我没有三颗心脏 一个拉胯的分库分表方案有多绝望?整个部门都在救火! - 冰河团队 动态规划入门必学之走方格问题 - Ofnoname PostgREST 与 PostgreSQL 角色权限配置全解析(生产级实践) - SheepDog1998 使用 UEFI 图形输出协议 GOP 在屏幕上显示图像的方法 - 阿源- Claude Code通关手册(五):组建你的AI专家团队,子代理系统 - 暮色之狐 一个程序员到架构师的催婚路之感悟(整整10年后的催婚相亲感悟) - MisterLip 用 Agent Skill 自动生成工作周报 - 赵康
高精度算法
myLv · 2026-06-17 · via 博客园_首页

一、为什么需要高精度运算

普通的整数类型(int、long long)都有固定的取值范围,例如64位 long long 最大只能存储约 9×10¹⁸ 的数值。当我们需要处理几十位、上百位甚至更长的大整数时(比如大数阶乘、密码学计算、超长数值运算等场景),内置类型会发生溢出,无法正确存储和计算。高精度运算的核心思路就是用字符串或数组存储大整数的每一位数字,模拟人类手算的方式完成加减乘除,从而突破数值范围的限制。

二、基本概念

高精度运算,本质是将大整数按位拆分,用字符串(或数组)保存每一位的数字,完全模拟小学阶段学习的竖式加减乘除规则,逐位计算并处理进位、借位。

  • 存储方式:通常用字符串保存,高位在前、低位在后(和日常书写习惯一致),在计算时会反转字符串,让下标 0 对应最低位,方便对齐处理。
  • 核心操作:进位处理(加法、乘法)、借位处理(减法)、前导零去除、数值大小比较。

三、实现方法

1)加法

运算原理

加法完全模拟小学竖式加法规则:

  1. 将两个数右对齐,从最低位开始逐位相加;
  2. 每一位的和 = 第一个数当前位 \(a[i]\) + 第二个数当前位 \(b[i]\) + 低位传来的进位 \(carry\)
  3. 当前位结果 = 和 % 10,向高位的进位 = 和 / 10;
  4. 所有位计算完成后,如果还有剩余进位,要在最高位补上。
// 代码中先将字符串反转,是为了让数组下标 0 对应最低位,天然实现右对齐,无需手动补位
string add(string a, string b)
{
    if (a.length() < b.length()) swap(a, b); // 大的数字放“上面”,方便操作
    reverse(a.begin(), a.end()), reverse(b.begin(), b.end());
    string res;
    int carry = 0;
    for (size_t i = 0; i < a.length(); ++i)
    {
        int x = a[i] - '0';
        int y = i < b.length() ? b[i] - '0' : 0;
        int sum = x + y + carry;
        res += (sum % 10) + '0';
        carry = sum / 10;
    }
    if (carry) res += carry + '0';
    reverse(res.begin(), res.end());
    return res;
}

2)减法

运算原理

减法同样模拟小学竖式减法规则:

  1. 先比较两个数的大小,用大数减小数,保证结果非负,最后再补上负号;
  2. 将两个数右对齐,从最低位开始逐位相减;
  3. 如果当前位被减数 < 减数 + 借位,就向高位借 1 当 10,同时标记借位;
  4. 计算完成后,结果会产生多余的前导零,需要去除。
为什么要去除前导零?

减法计算完成后,高位可能会出现连续的 0。例如计算 \(1000 - 999 = 1\),按位计算后反转得到的中间结果会是 \(0001\),前面的三个 0 就是前导零。前导零不符合数字的书写规范,也会影响后续运算的长度判断,因此必须去除多余的前导零,仅保留最后一个有效数字(注意结果为 0 时保留一个 0)。

string sub(string a, string b)
{
    bool swaped = false;
    if (a.length() < b.length() || (a.length() == b.length() && a < b))
    {
        swap(a, b);
        swaped = true;
    }
    reverse(a.begin(), a.end()), reverse(b.begin(), b.end());
    string res;
    int carry = 0;
    for (size_t i = 0; i < a.length(); ++i)
    {
        int x = a[i] - '0';
        int y = i < b.length() ? b[i] - '0' : 0;
        int sum = x - y - carry;
        if (sum < 0)
        {
            sum += 10;
            carry++;
        } else carry = 0;
        res += sum + '0';
    }
    /*-----去除前导零-----*/
    while (res.length() > 1 && res.back() == '0') res.pop_back();
    if (swaped) res += '-';
    reverse(res.begin(), res.end());
    return res;
}

3)乘法

运算原理

乘法模拟小学竖式乘法的错位相加规则:

  1. 第一个数的第 i 位,乘以第二个数的第 j 位,结果会落在结果的第 i+j 位上(最低位从 0 开始计数);
  2. 先把所有位的乘积累加到位对应的位置,暂不处理进位;
  3. 全部相乘完成后,统一从低位到高位处理进位,每一位满十进一;
  4. 最后去除高位多余的前导零,再转换为字符串。

注意:两个长度分别为 La 和 Lb 的数相乘,结果长度最多为 La+Lb 位。

string mul(string a, string b)
{
    int La = a.size(), Lb = b.size();
    vector<int> res(La + Lb, 0);
    reverse(a.begin(),a.end()), reverse(b.begin(),b.end());

    for (size_t i = 0; i < a.length(); i++)
    {
        int x = a[i] - '0';
        for (size_t j = 0; j < b.length(); j++)
        {
            int y = b[j] - '0';
            res[i+j] += x * y; // 把每次每位的相乘结果放在数组中,先不进位
        }
    }

    /*-----进位-----*/
    for (int i = 0; i < (int)res.size(); i++)
        if (res[i] >= 10)
        {
            res[i+1] += res[i] / 10;
            res[i] %= 10;
        }

    /*-----去除前导零-----*/
    while (res.size() > 1 && res.back() == 0) res.pop_back();

    string ret;
    for (int i = res.size()-1; i >= 0; i--) ret += res[i] + '0';

    return ret;
}

4)除法

运算原理

这里实现的是高精度整数除以普通 int 整数(高精度除以高精度复杂度更高,日常场景中高精度除低精度最常用),同样模拟竖式除法规则:

  1. 从被除数的最高位开始,逐位向下计算;
  2. 当前位数值 = 上一位的余数 × 10 + 当前位数字;
  3. 当前位的商 = 当前位数值 ÷ 除数,新的余数 = 当前位数值 % 除数;
  4. 逐位计算完成后,去除商的前导零,同时可以返回余数。

代码中提供了两种写法:一种返回商和余数的 pair,一种只返回商,核心逻辑完全一致。

// 高精度除法:a / b,a为大整数(字符串),b为int,返回商和余数
pair<string, int> div(string a, int b)
{
    string res;
    int remainder = 0; // 余数
    for (size_t i = 0; i < a.length(); ++i)
    {
        remainder = remainder * 10 + (a[i] - '0');
        res += (remainder / b) + '0';
        remainder %= b;
    }
    // 去除前导零
    size_t pos = res.find_first_not_of('0');
    if (pos == string::npos) res = "0";
    else res = res.substr(pos);

    return {res, remainder};
}

string divide(string a, int b)
{
    if (b == 0) return "Error: Division by zero";
    
    vector<int> num;
    for (char c : a) num.push_back(c - '0');
    
    vector<int> result;
    int remainder = 0;
    
    for (size_t i = 0; i < num.size(); i++)
    {
        int current = remainder * 10 + num[i];
        result.push_back(current / b);
        remainder = current % b;
    }
    
    // 去除前导零
    size_t start = 0;
    while (start < result.size() && result[start] == 0) start++;
    
    if (start == result.size()) return "0"; // 等于0时保留一个0
    
    string ret = "";
    for (size_t i = start; i < result.size(); i++) ret += result[i] + '0';
    
    return ret;
}

四、总结

高精度四则运算的核心思想高度统一:用字符串/数组拆分存储大整数的每一位,完全模拟人工竖式计算的规则

  • 加法、乘法:从低位到高位计算,重点处理进位;
  • 减法:从低位到高位计算,重点处理借位,注意正负号判断;
  • 除法(低精度除数):从高位到低位计算,逐位落数,保留余数。

所有运算最终都需要处理前导零,保证输出格式符合常规数字的书写习惯。这套实现可以处理任意长度的正整数运算,是算法竞赛和大数计算场景的基础工具。

五、最后碎碎念

由于学期接近期末,最近都在准备期末考试,所以更新慢了许多,还请见谅!此外如果你发现我的博客有错误的地方、语义矛盾或者有更好的方法,欢迎你的建议与指导!