游乐游手机版
首页/编程语言/文章详情

C++如何检测两个圆形是否相交 _ 游戏开发几何碰撞检测逻辑【干货】

时间:2026-05-01 10:38
C++如何检测两个圆形是否相交 | 游戏开发几何碰撞检测逻辑【干货】 判断两个圆是否相交,核心逻辑其实就一句话:圆心距的平方是否小于等于半径和的平方。如果严格区分,外离是圆心距平方大于半径和的平方,内含则是圆心距平方小于半径差的平方(这里假设第一个圆半径更大)。至于内切和外切这两种临界状态,在实际编

C++如何检测两个圆形是否相交 | 游戏开发几何碰撞检测逻辑【干货】

C++如何检测两个圆形是否相交 _ 游戏开发几何碰撞检测逻辑【干货】

判断两个圆是否相交,核心逻辑其实就一句话:圆心距的平方是否小于等于半径和的平方。如果严格区分,外离是圆心距平方大于半径和的平方,内含则是圆心距平方小于半径差的平方(这里假设第一个圆半径更大)。至于内切和外切这两种临界状态,在实际编程中必须引入一个容差ε来处理,否则浮点误差会让你抓狂。

用距离平方判断比开方更高效

从几何定义上看,两个圆相交的充要条件是圆心距离小于等于半径之和。但如果你在代码里老老实实地写 sqrt((x1-x2)*(x1-x2) + (y1-y2)*(y1-y2)),那就掉进性能陷阱了。浮点开方运算开销大,还会引入不必要的精度扰动。要知道,游戏循环里每帧可能要进行成百上千次这样的检测,sqrt 函数能不用就不用。

那怎么办?其实很简单,把不等式两边平方一下就行了。因为距离和半径都是非负数,平方操作不会改变不等号的方向。这样一来,判断逻辑就变成了:

dx = x1 - x2;
dy = y1 - y2;
r_sum = r1 + r2;
if (dx * dx + dy * dy <= r_sum * r_sum) {
    // 相交(含内切、外切、相交、包含)
}

这里有个细节需要注意:这个判断条件会把“一个圆完全包含在另一个圆内部”的情况也归类为“相交”。如果你的游戏逻辑需要严格区分“边缘相交”和“完全包含”,比如子弹必须碰到怪物边缘才算击中,被护盾完全包裹不算受伤,那就得在这个基础上再加一层判断。

区分“相交”“内含”“相离”三种状态

对于复杂的游戏逻辑,光知道“是否相交”往往不够。比如,粒子碰撞后需要反弹,这就得明确是“相交”;而判断一个角色是否被保护罩完全覆盖,则需要识别“内含”状态。这就需要我们把距离与半径的关系拆解得更细致一些:

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

  • dx*dx + dy*dy > (r1 + r2)*(r1 + r2) → 相离
  • dx*dx + dy*dy < (r1 - r2)*(r1 - r2)(这里假设 r1 >= r2)→ 内含(小圆完全在大圆内,无交点)
  • 其余情况 → 相交(含外切、内切、部分重叠)

这里有个关键点要特别注意:计算半径差的平方时,一定要用绝对值差。直接写 (r1 - r2)*(r1 - r2)r2 > r1 时会得到负值,导致判断出错。稳妥的写法是用 abs(r1 - r2) 取绝对值后再平方,或者直接比较 dx*dx + dy*dy((r1 > r2) ? (r1 - r2) : (r2 - r1)) * ... 的结果。

浮点误差下如何处理“恰好相切”

理论上,外切的条件是圆心距严格等于半径和。但在浮点数的世界里,distance == r1 + r2 这种精确相等几乎不可能发生。如果你用 == 来判断相切,那结果永远是“否”。

正确的做法是引入一个容差(epsilon)。如果你确实需要识别“近似相切”的状态(比如触发一个特殊的接触音效),可以这样判断:

float dist_sq = dx*dx + dy*dy;
float r_sum = r1 + r2;
float diff = fabsf(sqrtf(dist_sq) - r_sum); // 这里 sqrt 不可避免,但仅在极少数判定时用
if (diff < 1e-4f) { /* 近似外切 */ }

不过,更推荐的做法是全程使用平方比较,并设置一个容差区间:

float r_sum_sq = (r1 + r2) * (r1 + r2);
float eps = 1e-6f;
if (dist_sq > r_sum_sq && dist_sq < r_sum_sq + eps) { /* 视为刚接触 */ }

话说回来,对于大多数游戏物理碰撞而言,并不需要如此精细地区分相切状态。只要进入“相交”分支就触发碰撞响应,把相切视为相交的一种临界情况来处理,通常就足够了。

结构体封装与成员函数建议

最后,别再用一堆零散的 x, y, r 变量了。定义一个 struct Circle 结构体,并提供一个 intersects(const Circle& other) const 方法,代码的清晰度和复用性会立刻提升一个档次:

struct Circle {
    float x, y, r;
    bool intersects(const Circle& o) const {
        float dx = x - o.x;
        float dy = y - o.y;
        float r_sum = r + o.r;
        return dx*dx + dy*dy <= r_sum * r_sum;
    }
};

如果你的项目后期需要考虑使用SIMD指令集,或者需要进行大批量的碰撞检测(比如成千上万的子弹对敌人),那么可以将数据结构从AoS(数组结构体)转换为SoA(结构体数组)以优化性能。但对于单次检测,没必要过早优化。

真正容易被忽略,却可能引发诡异Bug的细节是:**半径必须是非负数**。如果构造函数不小心传入了一个负半径,r_sum 的计算就会出错,编译器不会报错,但运行时的行为将不可预测。一个良好的习惯是在构造函数或设置函数中加入 assert(r >= 0) 进行断言检查。

来源:https://www.php.cn/faq/2400081.html
上一篇如何在Python中实现PyTorch的Transformer架构_调用nn.Transformer模块 下一篇如何利用 CopyOnWriteArrayList 的读写分离机制实现在高频读、极低频写场景下的无锁化访问
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
PyTorch中使用多维索引张量对高维张量批量索引的正确方法
编程语言 · 2026-07-03

PyTorch中使用多维索引张量对高维张量批量索引的正确方法

本文深入讲解如何在 PyTorch 中利用形状为 [b, k] 的索引张量 B,对形状为 [b, m, n] 的高维张量 A 执行高效批量索引,最终得到 [b, k, n] 的输出。核心思路在于合理扩展索引维度并配合 torch gather 实现精准的逐行抽取。 很多人处理高维张量的批量索引时都会

Go中...操作符解包切片传递可变参数函数
编程语言 · 2026-07-03

Go中...操作符解包切片传递可变参数函数

在 Go 语言中,` ` 运算符放在切片变量后面(如 `slice `)的作用是将该切片“展开”为多个独立参数,专门用于调用那些接受可变参数(` T`)的函数,例如 `append` 或 `fmt Println`。这是一种类型安全的语法糖,并非省略号或通配符,能够帮助开发者更简洁地处理

macOS与WSL2下PHP多版本切换失效问题排查与修复指南
编程语言 · 2026-07-03

macOS与WSL2下PHP多版本切换失效问题排查与修复指南

本文深入分析在 macOS 或 WSL2(Ubuntu)开发环境中,通过 Homebrew 管理 PHP 多版本时,php -v 始终显示旧版本(如 php@5 6)的深层原因,并给出系统性解决方案,覆盖 PATH 冲突、符号链接逻辑、Shell 初始化配置、系统残留配置等关键环节。 遇到这种情况的

PHP JSON解析深层嵌套对象属性访问失败的解决方法
编程语言 · 2026-07-03

PHP JSON解析深层嵌套对象属性访问失败的解决方法

使用 json_decode() 解析 API 返回的 JSON 数据时,经常遇到某个子属性无法正常获取,始终返回 NULL —— 这是许多 PHP 开发者都曾碰到过的棘手问题。通常并非数据丢失,而是对象嵌套层级比预期更深,导致访问路径不正确。 举例来说,你看到返回的 JSON 里有一个 appea

nnU-Net v2预处理卡死问题的成因分析与实用解决指南
编程语言 · 2026-07-03

nnU-Net v2预处理卡死问题的成因分析与实用解决指南

> 使用 nnUNetv2_plan_and_preprocess 处理大规模数据集(例如 704 例样本)时,程序常因多进程加载导致死锁而停滞。核心原因在于默认并发数过高引发资源竞争或 I O 阻塞,适当降低并发数即可稳定完成全量预处理。 你在使用 `nnunetv2_plan_and_prepr