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

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

相关文章

判断PyTorch是GPU版还是CPU版的方法小结

《判断PyTorch是GPU版还是CPU版的方法小结》PyTorch作为当前最流行的深度学习框架之一,支持在CPU和GPU(NVIDIACUDA)上运行,所以对于深度学习开发者来说,正确识别PyTor... 目录前言为什么需要区分GPU和CPU版本?性能差异硬件要求如何检查PyTorch版本?方法1:使用命

Python如何精准判断某个进程是否在运行

《Python如何精准判断某个进程是否在运行》这篇文章主要为大家详细介绍了Python如何精准判断某个进程是否在运行,本文为大家整理了3种方法并进行了对比,有需要的小伙伴可以跟随小编一起学习一下... 目录一、为什么需要判断进程是否存在二、方法1:用psutil库(推荐)三、方法2:用os.system调用

Python实现特殊字符判断并去掉非字母和数字的特殊字符

《Python实现特殊字符判断并去掉非字母和数字的特殊字符》在Python中,可以通过多种方法来判断字符串中是否包含非字母、数字的特殊字符,并将这些特殊字符去掉,本文为大家整理了一些常用的,希望对大家... 目录1. 使用正则表达式判断字符串中是否包含特殊字符去掉字符串中的特殊字符2. 使用 str.isa

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

Python中判断对象是否为空的方法

《Python中判断对象是否为空的方法》在Python开发中,判断对象是否为“空”是高频操作,但看似简单的需求却暗藏玄机,从None到空容器,从零值到自定义对象的“假值”状态,不同场景下的“空”需要精... 目录一、python中的“空”值体系二、精准判定方法对比三、常见误区解析四、进阶处理技巧五、性能优化

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:输出结果示例代码结语作