成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

解析幾何:計算兩條線段的交點

開發 前端
求兩線段的交點,本質就是解方程,需要用到克萊姆法則,計算出來的交點是直線交點,不一定是線段交點,需要再判斷點是否在線段范圍內。

大家好,我是前端西瓜哥。

今天來實現計算兩條線段的交點的解析幾何算法。

我們要實現 getLineSegIntersection 方法:提供兩條線段,計算它們的交點。

每條線段會用兩個點坐標表示。

const getLineSegIntersection = (p1, p2, p3, p4) => {
  // 待實現
}

// 測試用例
getLineSegIntersection(
  { x: 1, y: 1 }, { x: 4, y: 4 },
  { x: 1, y: 4 }, { x: 4, y: 1 }
);
// 期望 { x: 2.5, y: 2.5 }

思路

思路很簡單,就是解兩條直線對應的一個二元一次方程組,求出 x 和 y。

如果無解或多解,說明直線平行,交點不存在。

如果有解,可拿到唯一交點,但也只能說明直線有交點,還需要判斷線段是否有交點。

所以我們需要判斷交點是否在線段的區間上。如果是,說明兩線段有交點,返回交點。

克拉姆法則

解方程組需要用到 克拉姆法則。

對于:

可轉換為矩陣形式表示:

然后計算主矩陣(最左邊的矩陣)的行列式,對角相乘然后相減:

如果行列式為 0,說明沒有唯一解。

如果不為 0,則有唯一解:

回到我們的兩條直線,我們用兩點式表示直線:

轉換成 Ax+By=C 的格式,得到:

于是:

const a = y2 - y1;
const b = x1 - x2;
const c = x1 * y2 - x2 * y1;

第二條線段同理:

const d = y4 - y3;
const e = x3 - x4;
const f = x3 * y4 - x4 * y3;

算法實現

interface Point {
  x: number;
  y: number;
}

const getLineSegIntersection = (
  p1: Point,
  p2: Point,
  p3: Point,
  p4: Point
): Point | null => {
  const { x: x1, y: y1 } = p1;
  const { x: x2, y: y2 } = p2;
  const { x: x3, y: y3 } = p3;
  const { x: x4, y: y4 } = p4;

  const a = y2 - y1;
  const b = x1 - x2;
  const c = x1 * y2 - x2 * y1;

  const d = y4 - y3;
  const e = x3 - x4;
  const f = x3 * y4 - x4 * y3;

  // 計算分母
  const denominator = a * e - b * d;

  // 判斷分母是否為 0(代表平行)
  if (Math.abs(denominator) < 0.000000001) {
    // 這里有個特殊的重疊但只有一個交點的情況,可以考慮處理一下
    return null;
  }

  const px = (c * e - f * b) / denominator;
  const py = (a * f - c * d) / denominator;

  // 判斷交點是否在兩個線段上
  if (
    px >= Math.min(x1, x2) &&
    px <= Math.max(x1, x2) &&
    py >= Math.min(y1, y2) &&
    py <= Math.max(y1, y2) &&
    px >= Math.min(x3, x4) &&
    px <= Math.max(x3, x4) &&
    py >= Math.min(y3, y4) &&
    py <= Math.max(y3, y4)
  ) {
    return { x: px, y: py };
  }

  return null;
};

變體

這個算法可以做一些變體,實現其他的算法。

變體1:兩線段是否有交點。

返回值換成布爾值即可。

變體2:計算兩直線的交點。

把判斷直線交點是否在線段上的邏輯去掉,然后直接返回點坐標即可。

優化點

重疊但卻只有一個交點的情況。

如果線段平行,有兩種情況:

  • 沒有重疊(0 個解)
  • 有部分重疊(多解)

如果部分重疊,可能有多個點,多個點的情況下也不知道拿哪個點作為交點好,這種情況下還是返回 null。

但有一個特殊的情況:重疊只有一個點(比如線段 a 的末點剛好是線段 b 的起點)。如果你的場景下判斷比較嚴格,你可以選擇返回這個點。要實現這部分也是有點點復雜的。

誤差處理。線段的兩個端點的距離非常小,計算出的結果也會非常小,可能會進入了 0 的絕對誤差范圍了,考慮改成相對誤差。

溢出風險。數值很大時有溢出風險,可以考慮計算一個縮放值,縮小后計算,計算完再放大回去。

結尾

總結一下,求兩線段的交點,本質就是解方程,需要用到克萊姆法則,計算出來的交點是直線交點,不一定是線段交點,需要再判斷點是否在線段范圍內。

不復雜,就是有一點點小細節。

責任編輯:姜華 來源: 前端西瓜哥
相關推薦

2023-08-30 07:34:41

2012-12-20 10:19:08

華為路由器接入設置

2023-04-05 14:31:19

Java計算移動

2019-04-04 13:36:25

云計算互聯網云廠商

2024-12-27 00:00:00

SQL死鎖數據庫

2011-06-07 11:21:34

路由負載

2009-06-19 15:25:13

ITSMNSM運維管理

2014-10-24 15:17:07

Android

2014-12-24 09:15:54

PaaS開源云服務

2012-05-11 13:15:12

戴爾虛擬化

2022-06-06 23:22:44

互聯網產品模式

2019-11-06 15:16:12

16GB8GB內存

2011-06-21 15:24:20

HP高密度計算服務器

2010-01-12 18:05:56

Linux Redha

2021-11-10 23:44:21

筆記本觸摸板技巧

2016-08-18 09:53:33

軟件定義存儲

2022-08-11 13:11:48

斯坦福大學英偉達VR 頭顯

2017-01-11 15:45:52

中國聯通光纜亞歐5號

2009-03-06 12:17:24

IBMPowerSystemPower6

2021-12-08 09:00:25

LeetCode容器算法
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美v在线 | 一二三在线视频 | 亚洲免费在线视频 | 3p视频在线观看 | 午夜免费视频 | 黄色精品| 欧美一级淫片免费视频黄 | 精品1区2区 | 久久一区二区三区电影 | 久久久久久久久久久久久9999 | 国产一区二区三区四区 | 亚洲国产精品日本 | 凹凸日日摸日日碰夜夜 | 亚洲成人免费电影 | 日韩三区在线 | 国产午夜精品福利 | 一级毛片免费完整视频 | 亚洲美女av网站 | 久久成人免费观看 | 男人天堂网址 | 日韩乱码在线 | 免费麻豆视频 | 超碰人人91 | 亚洲国产精品99久久久久久久久 | 奇米影视在线 | 亚洲精品久久久一区二区三区 | 一区二区av在线 | www.久久久 | 99re在线视频观看 | 中文字幕精品一区久久久久 | 大学生a级毛片免费视频 | 国产片侵犯亲女视频播放 | 欧美高清成人 | 亚洲精品视频在线 | 亚洲一区精品在线 | 国产91丝袜在线播放 | 成人精品国产免费网站 | 97超碰在线免费 | 在线四虎 | 国产福利在线看 | 国产精品美女久久久久久免费 |