0

0

C++中的std::midpoint和std::lerp算法?(数学计算中的中点与线性插值)

尼克

尼克

发布时间:2026-01-13 16:30:11

|

791人浏览过

|

来源于php中文网

原创

std::midpoint是C++20引入的安全中点计算函数,避免整数溢出和浮点精度丢失,仅支持同类型算术类型或指针,需包含且编译器支持C++20。

c++中的std::midpoint和std::lerp算法?(数学计算中的中点与线性插值)

std::midpoint 是什么,为什么不能直接用 (a + b) / 2

它计算两个同类型整数或浮点数的数学中点,但关键在于**避免溢出和精度丢失**。对 int 类型,a + b 可能溢出(比如 INT_MAX1 相加),而 std::midpoint(a, b) 内部用位运算或无溢出算术实现,安全得多。

使用场景:坐标计算、二分查找边界更新、图像像素索引平均等需要“中间位置”的地方。

  • 只接受相同算术类型(intintdoubledouble),不支持混合类型(如 intdouble
  • 对浮点数,它比 (a + b) * 0.5 更精确——尤其当 ab 量级差异极大时(例如 1e201.0
  • C++20 引入,需编译器支持(GCC 10+、Clang 11+、MSVC 19.28+),且包含

std::lerp 的行为和常见误用

std::lerp(a, b, t) 计算线性插值:a + t * (b - a),但它不是简单套公式——它被设计为**在 t ∈ [0,1] 时数值稳定、可预测、无爆炸误差**。

典型误用是传入超出 [0,1]t 值并期待“外推”,但标准不保证该情况下的精度或单调性;某些实现甚至对 t 或 t > 1 做特殊处理(如 clamping),行为依赖库实现。

立即学习C++免费学习笔记(深入)”;

无界AI
无界AI

一站式AI创作、搜索、分享服务

下载
  • tdouble(即使 abfloat),函数返回类型与 ab 相同
  • t == 0 时严格返回 a(非近似),t == 1 时严格返回 b,这对动画关键帧、插值校验很重要
  • ab 符号相反且绝对值极大时,b - a 可能损失精度,而 std::lerp 内部会规避这类问题(例如用分段策略)

对比示例:midpoint vs lerp vs 手写公式

#include 
#include 
#include 

int main() { int a = std::numeric_limits::max(); int b = std::numeric_limits::max() - 1;

// ❌ 危险:溢出未定义行为
// int bad_mid = (a + b) / 2;

// ✅ 安全中点
int safe_mid = std::midpoint(a, b); // 结果确定,无溢出

double x = 1e16;
double y = 1.0;
double t = 0.5;

// ❌ 手写可能精度崩坏
double naive = x + t * (y - x); // y - x ≈ -1e16,加法抵消严重

// ✅ lerp 保障精度
double lerped = std::lerp(x, y, t); // 更接近数学期望值

std::cout zuojiankuohaophpcnzuojiankuohaophpcn safe_mid zuojiankuohaophpcnzuojiankuohaophpcn "\n" zuojiankuohaophpcnzuojiankuohaophpcn lerped zuojiankuohaophpcnzuojiankuohaophpcn "\n";

}

实际使用要注意的兼容性和边界

这两个函数看似简单,但落地时容易忽略底层约束:

  • 它们都不支持用户自定义类型,仅限算术类型(intlong longfloatdouble 等)
  • std::midpoint 对指针也有效(C++20 允许),例如 std::midpoint(p, q) 给出指针中点,但要求 pq 指向同一数组,否则未定义
  • 若目标平台不支持 C++20,别硬上;可用 Boost.Math 的 boost::math::tools::midpoint 或手写安全版本(如 a + (b - a) / 2 对无符号整数,或位移对齐整数)
  • 调试时注意:GDB/LLDB 可能无法内联展开这些函数,单步时会跳过——别误以为逻辑没执行

最常被忽略的是类型一致性:传 shortint 会编译失败,连隐式转换都不给;必须显式转成同一类型再调用。

相关专题

更多
css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

558

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

98

2025.10.23

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

534

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

194

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

98

2025.10.23

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

0

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号