Javascript 如何检查两个给定的线段是否相交(How to check if two given line segments intersect)

embedded/2024/12/22 15:12:57/

 给定两条线段(p1, q1)和(p2, q2),判断给定的线段是否相交。
在讨论解决方案之前,让我们先定义方向的概念。平面中有序点三元组的方向可以是 
–逆时针 
–顺时针 
–共线 

下图显示了(a,b,c) 的不同可能方向

 方向在这里有什么用处? 
两条线段(p1,q1)和(p2,q2)相交,当且仅当以下两个条件之一得到验证时

1. 一般情况: 
– ( p1 , q1 , p2 ) 和 ( p1 , q1 , q2 ) 具有不同的方向,并且 
– ( p2 , q2 ,  p1 ) 和 ( p2 , q2 ,  q1 ) 具有不同的方向。

例子:  

2.特殊情况 
– ( p1 , q1 , p2 )、( p1 , q1 , q2 )、( p2 , q2 , p1 ) 和 ( p2 , q2 , q1 ) 均为共线,且 – ( p1 , q1 ) 和 ( p2 , q2 ) 
的 x 投影相交 – ( p1 , q1 ) 和 ( p2 , q2 )的 y 投影相交

例子:  

以下是基于上述想法的实现:

// Javascript program to check if two given line segments intersect 
  
class Point 

    constructor(x, y) 
    { 
        this.x = x; 
            this.y = y; 
    } 

  
// Given three collinear points p, q, r, the function checks if 
// point q lies on line segment 'pr' 
function onSegment(p, q, r) 

    if (q.x <= Math.max(p.x, r.x) && q.x >= Math.min(p.x, r.x) && 
        q.y <= Math.max(p.y, r.y) && q.y >= Math.min(p.y, r.y)) 
    return true; 
    
    return false; 

  
// To find orientation of ordered triplet (p, q, r). 
// The function returns following values 
// 0 --> p, q and r are collinear 
// 1 --> Clockwise 
// 2 --> Counterclockwise 
function orientation(p, q, r) 

  
    // See https://www.geeksforgeeks.org/orientation-3-ordered-points/ 
    // for details of below formula. 
    let val = (q.y - p.y) * (r.x - q.x) - 
            (q.x - p.x) * (r.y - q.y); 
    
    if (val == 0) return 0; // collinear 
    
    return (val > 0)? 1: 2; // clock or counterclock wise 

  
// The main function that returns true if line segment 'p1q1' 
// and 'p2q2' intersect. 
function doIntersect(p1, q1, p2, q2) 

  
    // Find the four orientations needed for general and 
    // special cases 
    let o1 = orientation(p1, q1, p2); 
    let o2 = orientation(p1, q1, q2); 
    let o3 = orientation(p2, q2, p1); 
    let o4 = orientation(p2, q2, q1); 
    
    // General case 
    if (o1 != o2 && o3 != o4) 
        return true; 
    
    // Special Cases 
    // p1, q1 and p2 are collinear and p2 lies on segment p1q1 
    if (o1 == 0 && onSegment(p1, p2, q1)) return true; 
    
    // p1, q1 and q2 are collinear and q2 lies on segment p1q1 
    if (o2 == 0 && onSegment(p1, q2, q1)) return true; 
    
    // p2, q2 and p1 are collinear and p1 lies on segment p2q2 
    if (o3 == 0 && onSegment(p2, p1, q2)) return true; 
    
    // p2, q2 and q1 are collinear and q1 lies on segment p2q2 
    if (o4 == 0 && onSegment(p2, q1, q2)) return true; 
    
    return false; // Doesn't fall in any of the above cases 

  
// Driver code 
let p1 = new Point(1, 1); 
let q1 = new Point(10, 1); 
let p2 = new Point(1, 2); 
let q2 = new Point(10, 2); 
  
if(doIntersect(p1, q1, p2, q2)) 
    document.write("Yes<br>"); 
else
    document.write("No<br>"); 
  
p1 = new Point(10, 1); q1 = new Point(0, 10); 
p2 = new Point(0, 0); q2 = new Point(10, 10); 
if(doIntersect(p1, q1, p2, q2)) 
    document.write("Yes<br>"); 
else
    document.write("No<br>"); 
  
p1 = new Point(-5, -5); q1 = new Point(0, 0); 
p2 = new Point(1, 1); q2 = new Point(10, 10);; 
if(doIntersect(p1, q1, p2, q2)) 
    document.write("Yes<br>"); 
else
    document.write("No<br>"); 
  
// This code is contributed by avanitrachhadiya2155  

 输出: 

No
Yes
No

时间复杂度: O(1)

空间复杂度:O(1)

资料来源: 
http://www.dcs.gla.ac.uk/~pat/52233/slides/Geometry1x1.pdf   

《算法导论》第三版,作者:Clifford Stein、Thomas H. Cormen、Charles E. Leiserson、Ronald L. Rivest


http://www.ppmy.cn/embedded/111625.html

相关文章

[杂项]pugi::xml获取xml中的注释节点

前言 想到学习xml时的一句话&#xff0c;xml中注释也会被算作一个节点。那么我们就可以通过 pugixml 把注释节点获取出来&#xff0c; <?xml version"1.0"?> <mesh name"mesh_root"><!--这是一个注释节点-->some text<![CDATA[so…

新书速览|JavaScript前端开发与实例教程(微课视频版)(第2版)

《JavaScript前端开发与实例教程&#xff08;微课视频版&#xff09;&#xff08;第2版&#xff09;》 本书内容 JavaScript是开发Web前端必须掌握的编程语言&#xff0c;《JavaScript前端开发案例教程&#xff1a;微课视频版》以真实的项目需求为导向&#xff0c;循序渐进、深…

TestCraft - GPT支持的测试想法生成器和自动化测试生成器

在当今快速变化的软件开发世界中&#xff0c;自动化测试已成为确保软件质量的关键环节。而随着AI技术的进步&#xff0c;越来越多的工具开始引入人工智能&#xff0c;来辅助生成测试用例和自动化测试脚本。其中&#xff0c;TestCraft&#xff0c;作为一款GPT支持的测试想法生成…

19. 删除链表的倒数第 N 个结点【 力扣(LeetCode) 】

零、LeetCode 原题 19. 删除链表的倒数第 N 个结点 一、题目描述 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 进阶&#xff1a;你能尝试使用一趟扫描实现吗&#xff1f; 二、测试用例 示例 1&#xff1a; 输入&#xff1a;hea…

工作地址和工具连接

1&#xff1a;原型图的作画 &#x1f389; 原型操作指南 (modao.cc)墨刀 - 个人空间 (modao.cc) ProcessOn - 我的文件 Axure

appium server gui详细按照步骤

1.安装appium server desktop Appium安装提供两种方式:桌面版和命令行版。其中桌面版又分为 Appium GuI 和 Appium Desktop 。作为初学者&#xff0c;用桌面版&#xff0c;对初学者比较友好。 官网下载地址&#xff1a;Releases appium/appium-desktop GitHubTags appium/…

Nginx 学习笔记

Nginx 简介 Nginx 是一个高性能的 HTTP 和反向代理服务器&#xff0c;特点是占有内存少和并发能力强。 Nginx 可以作为静态页面的 Web 服务器&#xff0c;同时还支持 CGI 协议的动态语言&#xff0c;比如 perl&#xff0c;php 等。但不支持 Java。Java 程序只能通过和tomcat配…

Oracle中VARCHAR和VARCHAR2的区别

Oracle中VARCHAR和VARCHAR2的区别 VARCHAR2 默认类型&#xff0c;VARCHAR2是Oracle中最常见的可变长度字符串类型VARCHAR2不区分NULL和空字符串&#xff0c;这意味着NULL和空字符串在VARCHAR2类型中被视为相同的值最大长度为4000字节&#xff08;从Oracle 12c开始&#xff0c…