偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

解析幾何:計(jì)算兩條線(xiàn)段的交點(diǎn)

開(kāi)發(fā) 前端
求兩線(xiàn)段的交點(diǎn),本質(zhì)就是解方程,需要用到克萊姆法則,計(jì)算出來(lái)的交點(diǎn)是直線(xiàn)交點(diǎn),不一定是線(xiàn)段交點(diǎn),需要再判斷點(diǎn)是否在線(xiàn)段范圍內(nèi)。

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

今天來(lái)實(shí)現(xiàn)計(jì)算兩條線(xiàn)段的交點(diǎn)的解析幾何算法。

我們要實(shí)現(xiàn) getLineSegIntersection 方法:提供兩條線(xiàn)段,計(jì)算它們的交點(diǎn)。

每條線(xiàn)段會(huì)用兩個(gè)點(diǎn)坐標(biāo)表示。

const getLineSegIntersection = (p1, p2, p3, p4) => {
  // 待實(shí)現(xiàn)
}

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

思路

思路很簡(jiǎn)單,就是解兩條直線(xiàn)對(duì)應(yīng)的一個(gè)二元一次方程組,求出 x 和 y。

如果無(wú)解或多解,說(shuō)明直線(xiàn)平行,交點(diǎn)不存在。

如果有解,可拿到唯一交點(diǎn),但也只能說(shuō)明直線(xiàn)有交點(diǎn),還需要判斷線(xiàn)段是否有交點(diǎn)。

所以我們需要判斷交點(diǎn)是否在線(xiàn)段的區(qū)間上。如果是,說(shuō)明兩線(xiàn)段有交點(diǎn),返回交點(diǎn)。

克拉姆法則

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

對(duì)于:

可轉(zhuǎn)換為矩陣形式表示:

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

如果行列式為 0,說(shuō)明沒(méi)有唯一解。

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

回到我們的兩條直線(xiàn),我們用兩點(diǎn)式表示直線(xiàn):

轉(zhuǎn)換成 Ax+By=C 的格式,得到:

于是:

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

第二條線(xiàn)段同理:

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

算法實(shí)現(xiàn)

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;

  // 計(jì)算分母
  const denominator = a * e - b * d;

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

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

  // 判斷交點(diǎn)是否在兩個(gè)線(xiàn)段上
  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;
};

變體

這個(gè)算法可以做一些變體,實(shí)現(xiàn)其他的算法。

變體1:兩線(xiàn)段是否有交點(diǎn)。

返回值換成布爾值即可。

變體2:計(jì)算兩直線(xiàn)的交點(diǎn)。

把判斷直線(xiàn)交點(diǎn)是否在線(xiàn)段上的邏輯去掉,然后直接返回點(diǎn)坐標(biāo)即可。

優(yōu)化點(diǎn)

重疊但卻只有一個(gè)交點(diǎn)的情況。

如果線(xiàn)段平行,有兩種情況:

  • 沒(méi)有重疊(0 個(gè)解)
  • 有部分重疊(多解)

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

但有一個(gè)特殊的情況:重疊只有一個(gè)點(diǎn)(比如線(xiàn)段 a 的末點(diǎn)剛好是線(xiàn)段 b 的起點(diǎn))。如果你的場(chǎng)景下判斷比較嚴(yán)格,你可以選擇返回這個(gè)點(diǎn)。要實(shí)現(xiàn)這部分也是有點(diǎn)點(diǎn)復(fù)雜的。

誤差處理。線(xiàn)段的兩個(gè)端點(diǎn)的距離非常小,計(jì)算出的結(jié)果也會(huì)非常小,可能會(huì)進(jìn)入了 0 的絕對(duì)誤差范圍了,考慮改成相對(duì)誤差。

溢出風(fēng)險(xiǎn)。數(shù)值很大時(shí)有溢出風(fēng)險(xiǎn),可以考慮計(jì)算一個(gè)縮放值,縮小后計(jì)算,計(jì)算完再放大回去。

結(jié)尾

總結(jié)一下,求兩線(xiàn)段的交點(diǎn),本質(zhì)就是解方程,需要用到克萊姆法則,計(jì)算出來(lái)的交點(diǎn)是直線(xiàn)交點(diǎn),不一定是線(xiàn)段交點(diǎn),需要再判斷點(diǎn)是否在線(xiàn)段范圍內(nèi)。

不復(fù)雜,就是有一點(diǎn)點(diǎn)小細(xì)節(jié)。

責(zé)任編輯:姜華 來(lái)源: 前端西瓜哥
相關(guān)推薦

2023-08-30 07:34:41

2012-12-20 10:19:08

華為路由器接入設(shè)置

2023-04-05 14:31:19

Java計(jì)算移動(dòng)

2019-04-04 13:36:25

云計(jì)算互聯(lián)網(wǎng)云廠(chǎng)商

2024-12-27 00:00:00

SQL死鎖數(shù)據(jù)庫(kù)

2011-06-07 11:21:34

路由負(fù)載

2009-06-19 15:25:13

ITSMNSM運(yùn)維管理

2014-10-24 15:17:07

Android

2014-12-24 09:15:54

PaaS開(kāi)源云服務(wù)

2012-05-11 13:15:12

戴爾虛擬化

2022-06-06 23:22:44

互聯(lián)網(wǎng)產(chǎn)品模式

2019-11-06 15:16:12

16GB8GB內(nèi)存

2011-06-21 15:24:20

HP高密度計(jì)算服務(wù)器

2010-01-12 18:05:56

Linux Redha

2021-11-10 23:44:21

筆記本觸摸板技巧

2016-08-18 09:53:33

軟件定義存儲(chǔ)

2022-08-11 13:11:48

斯坦福大學(xué)英偉達(dá)VR 頭顯

2017-01-11 15:45:52

中國(guó)聯(lián)通光纜亞歐5號(hào)

2009-03-06 12:17:24

IBMPowerSystemPower6

2021-12-08 09:00:25

LeetCode容器算法
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)