判断两个矩形是否有重合部分

2024-06-05 14:38

本文主要是介绍判断两个矩形是否有重合部分,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在做人工智能项目,需要对两个矩形是否有重合做出判读

但注意的是,不是判断两个检测目标是否重合,检测目标的矩形只要左上角点和右下角点就可表示一个矩形,判断是否重合比较简单,但是现在是两个矩形是有一定旋转角度的,旋转角度不定,这样就没法像检测目标那样判断了。

目前使用四个点坐标表示一个矩形

判断分两个步骤:

一、当一个矩形的顶点在另一矩形内时,则可以判断两个矩形有重合。当这种情况不存在时,不能判断两个矩形不重合,需要借助其他方法判断。

判断矩形任意一顶点是否在另一矩形内,这样就需要利用点在多边形内的判断方法

可以参考博客https://blog.csdn.net/gf771115/article/details/42870605/,这篇博客对原理做了较为详细的讲解,并且提供了判断函数的不同演进形式。

int pnploy(const int poly_sides, const float *poly_X, const float *poly_Y, const float x, const float y)
{
     int i, j;
     j = poly_sides - 1;
     int res = 0;
     for (i = 0; i<poly_sides; i++)
     {
         if ((poly_Y[i]<y && poly_Y[j] >= y || poly_Y[j]<y && poly_Y[i] >= y) && (poly_X[i] <= x || poly_X[j] <= x))
         {
             res ^= ((poly_X[i] + (y - poly_Y[i]) / (poly_Y[j] - poly_Y[i])*(poly_X[j] - poly_X[i])) < x);
         }
         j = i;
     }
     return res;
}

该判断方法又叫射线法,由点向任何一个方向引出射线,与多边形交点数为奇数的,则点在多边形内,为偶数时,则点不在多边形内,为了简化这种判断过程,通常C++代码中使用水平射线来做计算。

二、当任一矩形的任何顶点都不在另一矩形内时,是不能判断两个矩形不重合的,因为还有一些特殊情况

特例如下,以下情况就是顶点不在另一个矩形内,但是矩形重合的。

这样就需要利用矩形边交叉来判断重合,通常矩形边交叉,则相互重合的。

这样就需要判断两条线段是否交叉,

https://www.cnblogs.com/wuwangchuxin0924/p/6218494.html 该博客为我提供了方法

判断线段交叉两个步骤,(1)快速排斥 (2)跨立实验,具体请参考我上面提供的链接
bool linecross(const Point a, const Point b, const Point c,const Point d)
{
    //快速排斥
    if(!(std::min(a.x,b.x)<=std::max(c.x,d.x)
         && std::min(c.x,d.x)<= std::max(a.x,b.x)
         && std::min(a.y,b.y)<= std::max(c.y,d.y)
         && std::min(c.y,d.y)<= std::max(a.y,b.y)))
        return false;
    //跨立实验
    double u,v,w,z;
    u=(c.x-a.x)*(b.y-a.y)-(b.x-a.x)*(c.y-a.y);  //AC×AB
    v=(d.x-a.x)*(b.y-a.y)-(b.x-a.x)*(d.y-a.y);  //AD×AB
    w=(a.x-c.x)*(d.y-c.y)-(d.x-c.x)*(a.y-c.y);  //CA×AB
    z=(b.x-c.x)*(d.y-c.y)-(d.x-c.x)*(b.y-c.y);  //CB×AB

    if(u*v<=1e-9&&w*z<=1e-9)
        return true;
    return false;
}

 

好,这样我们就可以基于以上提供的方法,做出两个矩形是否会重合的判断逻辑了,目前可以试用于四边形,没测试过其他形状。大家如果觉得判断有问题或者一些效率上的问题,欢迎提出。
//目前只支持4边形交互
bool checkployintersect(const Point ploy1[4], const Point ploy2[4])
{
    Polygon polygon1, polygon2;
    for(int i = 0; i< 4; i++)
    {
        polygon1.x[i]=ploy1[i].x;
        polygon1.y[i]=ploy1[i].y;
        polygon2.x[i]=ploy2[i].x;
        polygon2.y[i]=ploy2[i].y;
    }

    for(int i =0; i<4; i++)
    {
        if(pnploy(4,polygon1.x,polygon1.y, polygon2.x[i], polygon2.y[i]))
        {
            return true;
        }
        if(pnploy(4,polygon2.x,polygon2.y, polygon1.x[i], polygon1.y[i]))
        {
            return true;
        }
    }
    for(int i=0; i<4; i++)
    {
        for(int j=0; j<4; j++)
        {
            if(linecross(ploy1[i%4], ploy1[(i+1)%4], ploy2[j%4], ploy2[(j+1)%4]))
            {
                return true;
            }
        }
    }
    return false;
}

 

这篇关于判断两个矩形是否有重合部分的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1033336

相关文章

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

java两个List的交集,并集方式

《java两个List的交集,并集方式》文章主要介绍了Java中两个List的交集和并集的处理方法,推荐使用Apache的CollectionUtils工具类,因为它简单且不会改变原有集合,同时,文章... 目录Java两个List的交集,并集方法一方法二方法三总结java两个List的交集,并集方法一

C++实现回文串判断的两种高效方法

《C++实现回文串判断的两种高效方法》文章介绍了两种判断回文串的方法:解法一通过创建新字符串来处理,解法二在原字符串上直接筛选判断,两种方法都使用了双指针法,文中通过代码示例讲解的非常详细,需要的朋友... 目录一、问题描述示例二、解法一:将字母数字连接到新的 string思路代码实现代码解释复杂度分析三、

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

Python如何计算两个不同类型列表的相似度

《Python如何计算两个不同类型列表的相似度》在编程中,经常需要比较两个列表的相似度,尤其是当这两个列表包含不同类型的元素时,下面小编就来讲讲如何使用Python计算两个不同类型列表的相似度吧... 目录摘要引言数字类型相似度欧几里得距离曼哈顿距离字符串类型相似度Levenshtein距离Jaccard相

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

C#比较两个List集合内容是否相同的几种方法

《C#比较两个List集合内容是否相同的几种方法》本文详细介绍了在C#中比较两个List集合内容是否相同的方法,包括非自定义类和自定义类的元素比较,对于非自定义类,可以使用SequenceEqual、... 目录 一、非自定义类的元素比较1. 使用 SequenceEqual 方法(顺序和内容都相等)2.

查询Oracle数据库表是否被锁的实现方式

《查询Oracle数据库表是否被锁的实现方式》本文介绍了查询Oracle数据库表是否被锁的方法,包括查询锁表的会话、人员信息,根据object_id查询表名,以及根据会话ID查询和停止本地进程,同时,... 目录查询oracle数据库表是否被锁1、查询锁表的会话、人员等信息2、根据 object_id查询被