谈谈"求线段交点"的几种算法

 我来答
世纪网络17
2022-10-29 · TA获得超过5884个赞
知道小有建树答主
回答量:2426
采纳率:100%
帮助的人:135万
展开全部

  求线段交点 是一种非常基础的几何计算 在很多游戏中都会被使用到

  下面我就现学现卖的把最近才学会的一些 求线段交点 的算法说一说 希望对大家有所帮助

  本文讲的内容都很初级 主要是面向和我一样的初学者 所以请各位算法帝们轻拍啊 嘎嘎

  引用已知线段 (a b) 和线段 (c d) 其中a b c d为端点 求线段交点p (平行或共线视作不相交)

  算法一 求两条线段所在直线的交点 再判断交点是否在两条线段上

  求直线交点时 我们可通过直线的一般方程 ax+by+c= 求得(方程中的abc为系数 不是前面提到的端点 另外也可用点斜式方程和斜截式方程 此处暂且不论)

  然后根据交点的与线段端点的位置关系来判断交点是否在线段上 公式如下图

    

【责编:at 】

  算法二 判断每一条线段的两个端点是否都在另一条线段的两侧 是则求出两条线段所在直线的交点 否则不相交

  第一步判断两个点是否在某条线段的两侧 通常可采用投影法

  求出线段的法线向量 然后把点投影到法线上 最后根据投影的位置来判断点和线段的关系 见下图

   

  点a和点b在线段cd法线上的投影如图所示 这时候我们还要做一次线段cd在自己法线上的投影(选择点c或点d中的一个即可)

  主要用来做参考

  图中点a投影和点b投影在点c投影的两侧 说明线段ab的端点在线段cd的两侧

  同理 再判断一次cd是否在线段ab两侧即可

  求法线 求投影 什么的听起来很复杂的样子 实际上对于我来说也确实挺复杂 在几个月前我也不会(念书那会儿的几何知识都忘光了 ( ) 不过好在学习和实现起来还不算复杂 皆有公式可循

  求线段ab的法线

  JavaScript Code复制内容到剪贴板var nx=b y a y ny=a x b x var normalLine = {  x nx y ny } 注意 其中 normalLine x和normalLine y的几何意义表示法线的方向 而不是坐标

  求点c在法线上的投影位置

  JavaScript Code复制内容到剪贴板var dist= normalLine x*c x + normalLine y*c y

  注意 这里的 投影位置 是一个标量 表示的是到法线原点的距离 而不是投影点的坐标

  通常知道这个距离就足够了

  当我们把图中 点a投影(distA) 点b投影(distB) 点c投影(distC) 都求出来之后 就可以很容易的根据各自的大小判断出相对位置

  distA==distB==distC 时 两条线段共线distA==distB!=distC 时 两条线段平行distA 和 distB 在distC 同侧时 两条线段不相交

  distA 和 distB 在distC 异侧时 两条线段是否相交需要再判断点c点d与线段ab的关系

  前面的那些步骤 只是实现了 判断线段是否相交 当结果为true时 我们还需要进一步求交点

  求交点的过程后面再说 先看一下该算法的完整实现

  JavaScript Code复制内容到剪贴板function segmentsIntr(a b c d){

  //线段ab的法线N var nx = (b y a y) ny = (a x b x)

  //线段cd的法线N var nx = (d y c y) ny = (c x d x)

  //两条法线做叉乘 如果结果为 说明线段ab和线段cd平行或共线 不相交var denominator = nx *ny ny *nx if (denominator== ) { return false }

  //在法线N 上的投影var distC_N =nx * c x + ny * c y var distA_N =nx * a x + ny * a y distC_N var distB_N =nx * b x + ny * b y distC_N

  // 点a投影和点b投影在点c投影同侧 (对点在线段上的情况 本例当作不相交处理) if ( distA_N *distB_N >=   ) { return false }

  // //判断点c点d 和线段ab的关系 原理同上// //在法线N 上的投影var distA_N =nx * a x + ny * a y var distC_N =nx * c x + ny * c y distA_N var distD_N =nx * d x + ny * d y distA_N if ( distC_N *distD_N >=   ) { return false }

  //计算交点坐标var fraction= distA_N / denominator var dx= fraction * ny dy= fraction * nx return { x a x + dx y a y + dy } }

  最后 求交点坐标的部分 所用的方法看起来有点奇怪 有种摸不著头脑的感觉

  其实它和算法一 里面的算法是类似的 只是里面的很多计算项已经被提前计算好了

  换句话说 算法二里求交点坐标的部分 其实也是用的直线的线性方程组来做的

  现在来简单粗略 很不科学的对比一下算法一和算法二 最好情况下 两种算法的复杂度相同 最坏情况 算法一和算法二的计算量差不多 但是算法二提供了 更多的 提前结束条件 所以平均情况下 应该算法二更优

  实际测试下来 实际情况也确实如此

【责编:at 】

  算法三 判断每一条线段的两个端点是否都在另一条线段的两侧 是则求出两条线段所在直线的交点 否则不相交

  (咦? 怎么感觉和算法二一样啊? 不要怀疑 确实一样 …… 囧)

  所谓算法三 其实只是对算法二的一个改良 改良的地方主要就是 不通过法线投影来判断点和线段的位置关系 而是通过点和线段构成的三角形面积来判断

  先来复习下三角形面积公式 已知三角形三点a(x y) b(x y) c(x y) 三角形面积为

  JavaScript Code复制内容到剪贴板var triArea=( (a x c x) * (b y c y) (a y c y) * (b x c x) ) /

  因为 两向量叉乘==两向量构成的平行四边形(以两向量为邻边)的面积 所以上面的公式也不难理解

  而且由于向量是有方向的 所以面积也是有方向的 通常我们以逆时针为正 顺时针为负数

  改良算法关键点就是 如果 线段ab和点c构成的三角形面积 与 线段ab和点d构成的三角形面积 构成的三角形面积的正负符号相异 那么点c和点d位于线段ab两侧 如下图所示

【责编:at 】 lishixinzhi/Article/program/Java/Javascript/201311/25424

已赞过 已踩过<
你对这个回答的评价是?
评论 收起
上海华然企业咨询
2024-10-21 广告
上海华然企业咨询有限公司专注于AI与数据合规咨询服务。我们的核心团队来自头部互联网企业、红圈律所和专业安全服务机构。凭借深刻的AI产品理解、上百个AI产品的合规咨询和算法备案经验,为客户提供专业的算法备案、AI安全评估、数据出境等合规服务,... 点击进入详情页
本回答由上海华然企业咨询提供
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式