已知圆弧的起点、终点、凸度 求圆弧的圆心

C#

csharp 复制代码
/// <summary>
/// 根据圆弧起点、终点和凸度(bulge)计算圆心坐标和半径
/// 无任何结构体依赖,纯 double 输入输出
/// </summary>
/// <param name="x1">起点 X</param>
/// <param name="y1">起点 Y</param>
/// <param name="x2">终点 X</param>
/// <param name="y2">终点 Y</param>
/// <param name="bulge">凸度(可正可负)</param>
/// <param name="centerX">输出:圆心 X</param>
/// <param name="centerY">输出:圆心 Y</param>
/// <param name="radius">输出:半径</param>
/// <returns>是否成功(true=成功,false=退化为点或直线)</returns>
public static bool GetArcCenter(
    double x1, double y1,
    double x2, double y2,
    double bulge,
    out double centerX, out double centerY,
    out double radius)
{
    double dx = x2 - x1;
    double dy = y2 - y1;
    double L = Math.Sqrt(dx * dx + dy * dy); // 弦长

    // 起点与终点重合,无法构成圆弧
    if (L < 1e-12)
    {
        centerX = 0;
        centerY = 0;
        radius = 0;
        return false;
    }

    // 矢高 h = (bulge * chord_length) / 2,带符号!
    double h = bulge * L * 0.5;

    // 凸度为0 → 是直线段,不是圆弧
    if (Math.Abs(h) < 1e-12)
    {
        centerX = 0;
        centerY = 0;
        radius = 0;
        return false;
    }

    // 圆心到弦中点的偏移距离 D(沿法线方向)
    double D = (L * L - 4 * h * h) / (8 * h);

    // 弦中点
    double midX = (x1 + x2) * 0.5;
    double midY = (y1 + y2) * 0.5;

    // 单位法向量(垂直于弦,逆时针方向): (-dy, dx) / L
    double nx = -dy / L;
    double ny = dx / L;

    // 圆心 = 中点 + D * 法向量
    centerX = midX + D * nx;
    centerY = midY + D * ny;

    // 半径 = 圆心到起点的距离
    double dxToCenter = centerX - x1;
    double dyToCenter = centerY - y1;
    radius = Math.Sqrt(dxToCenter * dxToCenter + dyToCenter * dyToCenter);

    return true;
}

C++

cpp 复制代码
#include <cmath>

/**
 * @brief 根据圆弧起点、终点和凸度(bulge)计算圆心和半径
 * @param x1 起点 X
 * @param y1 起点 Y
 * @param x2 终点 X
 * @param y2 终点 Y
 * @param bulge 凸度(可正可负)
 * @param centerX 输出:圆心 X
 * @param centerY 输出:圆心 Y
 * @param radius 输出:半径
 * @return bool 成功返回 true,失败(重合点或直线)返回 false
 */
bool GetArcCenter(
    double x1, double y1,
    double x2, double y2,
    double bulge,
    double& centerX, double& centerY,
    double& radius)
{
    double dx = x2 - x1;
    double dy = y2 - y1;
    double L = std::sqrt(dx * dx + dy * dy); // 弦长

    // 起点与终点重合,无法构成圆弧
    if (L < 1e-12)
    {
        centerX = 0.0;
        centerY = 0.0;
        radius = 0.0;
        return false;
    }

    // 矢高 h = (bulge * chord_length) / 2,带符号!
    double h = bulge * L * 0.5;

    // 凸度为0 → 是直线段,不是圆弧
    if (std::abs(h) < 1e-12)
    {
        centerX = 0.0;
        centerY = 0.0;
        radius = 0.0;
        return false;
    }

    // 圆心到弦中点的偏移距离 D(沿法线方向)
    double D = (L * L - 4 * h * h) / (8 * h);

    // 弦中点
    double midX = (x1 + x2) * 0.5;
    double midY = (y1 + y2) * 0.5;

    // 单位法向量(垂直于弦,逆时针方向): (-dy, dx) / L
    double nx = -dy / L;
    double ny = dx / L;

    // 圆心 = 中点 + D * 法向量
    centerX = midX + D * nx;
    centerY = midY + D * ny;

    // 半径 = 圆心到起点的距离
    double dxToCenter = centerX - x1;
    double dyToCenter = centerY - y1;
    radius = std::sqrt(dxToCenter * dxToCenter + dyToCenter * dyToCenter);

    return true;
}
相关推荐
無限進步D12 分钟前
蓝桥杯赛前刷题
c++·算法·蓝桥杯·竞赛
CoderCodingNo14 分钟前
【GESP】C++二级真题 luogu-B4497, [GESP202603 二级] 数数
开发语言·c++·算法
磊 子17 分钟前
八大排序之冒泡排序+选择排序
数据结构·算法·排序算法
We་ct18 分钟前
LeetCode 50. Pow(x, n):从暴力法到快速幂的优化之路
开发语言·前端·javascript·算法·leetcode·typescript·
潇洒畅想23 分钟前
1.1 从∑到∫:用循环理解求和与累积
java·数据结构·python·算法
郝学胜-神的一滴1 小时前
[简化版 GAMES 101] 计算机图形学 04:二维变换上
c++·算法·unity·godot·图形渲染·unreal engine·cesium
ZC跨境爬虫1 小时前
海南大学交友平台开发实战day7(实现核心匹配算法+解决JSON请求报错问题)
前端·python·算法·html·json
计算机安禾1 小时前
【数据结构与算法】第41篇:图论(五):拓扑排序与关键路径
c语言·数据结构·c++·算法·图论·visual studio
Q741_1471 小时前
每日一题 力扣 1320. 二指输入的的最小距离 动态规划 C++ 题解
c++·算法·leetcode·动态规划
wfbcg1 小时前
每日算法练习:LeetCode 76. 最小覆盖子串 ✅
算法·leetcode·职场和发展