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

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

相关文章

shell脚本快速检查192.168.1网段ip是否在用的方法

《shell脚本快速检查192.168.1网段ip是否在用的方法》该Shell脚本通过并发ping命令检查192.168.1网段中哪些IP地址正在使用,脚本定义了网络段、超时时间和并行扫描数量,并使用... 目录脚本:检查 192.168.1 网段 IP 是否在用脚本说明使用方法示例输出优化建议总结检查 1

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

poj 2976 分数规划二分贪心(部分对总体的贡献度) poj 3111

poj 2976: 题意: 在n场考试中,每场考试共有b题,答对的题目有a题。 允许去掉k场考试,求能达到的最高正确率是多少。 解析: 假设已知准确率为x,则每场考试对于准确率的贡献值为: a - b * x,将贡献值大的排序排在前面舍弃掉后k个。 然后二分x就行了。 代码: #include <iostream>#include <cstdio>#incl

zoj 1721 判断2条线段(完全)相交

给出起点,终点,与一些障碍线段。 求起点到终点的最短路。 枚举2点的距离,然后最短路。 2点可达条件:没有线段与这2点所构成的线段(完全)相交。 const double eps = 1e-8 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;

POJ1269 判断2条直线的位置关系

题目大意:给两个点能够确定一条直线,题目给出两条直线(由4个点确定),要求判断出这两条直线的关系:平行,同线,相交。如果相交还要求出交点坐标。 解题思路: 先判断两条直线p1p2, q1q2是否共线, 如果不是,再判断 直线 是否平行, 如果还不是, 则两直线相交。  判断共线:  p1p2q1 共线 且 p1p2q2 共线 ,共线用叉乘为 0  来判断,  判断 平行:  p1p

Codeforces Round #113 (Div. 2) B 判断多边形是否在凸包内

题目点击打开链接 凸多边形A, 多边形B, 判断B是否严格在A内。  注意AB有重点 。  将A,B上的点合在一起求凸包,如果凸包上的点是B的某个点,则B肯定不在A内。 或者说B上的某点在凸包的边上则也说明B不严格在A里面。 这个处理有个巧妙的方法,只需在求凸包的时候, <=  改成< 也就是说凸包一条边上的所有点都重复点都记录在凸包里面了。 另外不能去重点。 int

两个月冲刺软考——访问位与修改位的题型(淘汰哪一页);内聚的类型;关于码制的知识点;地址映射的相关内容

1.访问位与修改位的题型(淘汰哪一页) 访问位:为1时表示在内存期间被访问过,为0时表示未被访问;修改位:为1时表示该页面自从被装入内存后被修改过,为0时表示未修改过。 置换页面时,最先置换访问位和修改位为00的,其次是01(没被访问但被修改过)的,之后是10(被访问了但没被修改过),最后是11。 2.内聚的类型 功能内聚:完成一个单一功能,各个部分协同工作,缺一不可。 顺序内聚:

easyui同时验证账户格式和ajax是否存在

accountName: {validator: function (value, param) {if (!/^[a-zA-Z][a-zA-Z0-9_]{3,15}$/i.test(value)) {$.fn.validatebox.defaults.rules.accountName.message = '账户名称不合法(字母开头,允许4-16字节,允许字母数字下划线)';return fal