opnecv中的RotatedRect

在使用opencv的过程中经常会使用minAreaRect返回一个轮廓的外接矩形,它返回的是一个RotatedRect的类:

[html]  view plain  copy
  1. class CV_EXPORTS RotatedRect  
  2. {  
  3. public:  
  4.     //! various constructors  
  5.     RotatedRect();  
  6.     RotatedRect(const Point2f& center, const Size2f& size, float angle);  
  7.     RotatedRect(const CvBox2D& box);  
  8.   
  9.     //! returns 4 vertices of the rectangle  
  10.     void points(Point2f pts[]) const;  
  11.     //! returns the minimal up-right rectangle containing the rotated rectangle  
  12.     Rect boundingRect() const;  
  13.     //! conversion to the old-style CvBox2D structure  
  14.     operator CvBox2D() const;  
  15.   
  16.     Point2f center; //< the rectangle mass center  
  17.     Size2f size;    //< width and height of the rectangle  
  18.     float angle;    //< the rotation angle. When the angle is 0, 90, 180, 270 etc., the rectangle becomes an up-right rectangle.  
  19. };  

这个类中包含了外接矩形的中心center、大小size以及角度angle。为了更好的理解这几个参数的意义,请看下图:

在opencv中,坐标的原点在左上角,与x轴平行的方向为角度为0,逆时针旋转角度为负,顺时针旋转角度为正。而RotatedRect类是以矩形的哪一条边与x轴的夹角作为角度的呢?参考http://blog.csdn.net/mine1024/article/details/6044856的观点:angle 是水平轴(x轴)逆时针旋转,与碰到的第一个边的夹角,而opencv默认把这个边的边长作为width。由前面所说,angle的取值范围必然是负的,实际angle的取值范围为(-90,0]。利用成员函数void points(Point2f pts[]) const;可以计算出矩形的四个角点。计算的原理很简单:

图中的θ=-angle。从图中可知p[0].x=center.x-a

其中a=0.5*width*cosθ - b, b=heigth*sinθ,且cosθ=cos(angle),sinθ=sin(-angle)=-sin(angle)

那么p[0].x=center.x - 0.5*(width*cos(angle) + heigth*sin(angle)),对于p[0].y也可以用同样的原理计算,对应opencv中的源代码如下:

[cpp]  view plain  copy
  1. void RotatedRect::points(Point2f pt[]) const  
  2. {  
  3.     double _angle = angle*CV_PI/180.;  
  4.     float b = (float)cos(_angle)*0.5f;  
  5.     float a = (float)sin(_angle)*0.5f;  
  6.   
  7.     pt[0].x = center.x - a*size.height - b*size.width;  
  8.     pt[0].y = center.y + b*size.height - a*size.width;  
  9.     pt[1].x = center.x + a*size.height - b*size.width;  
  10.     pt[1].y = center.y - b*size.height - a*size.width;  
  11.     pt[2].x = 2*center.x - pt[0].x;  
  12.     pt[2].y = 2*center.y - pt[0].y;  
  13.     pt[3].x = 2*center.x - pt[1].x;  
  14.     pt[3].y = 2*center.y - pt[1].y;  
  15. }  

由此可知道,在opencv中,RotatedRect的角度实际上就是水平轴x与矩形宽width的夹角,而在利用minAreaRect函数去求一个外接矩形时,函数对于矩形长和宽的选择取决于这个外接矩形的边与水平轴的角度距离,即将水平轴逆时针旋转,最先与其平行的边的长度作为宽度width,另外一条边则为height,而角度则直接取该边与水平轴的夹角。应该意识到,在minAreaRect函数中,RotatedRect的width和height的选取跟矩形的尺寸无关,并非长的就是height,短的就是width。

原文地址

https://blog.csdn.net/a553654745/article/details/45743063

微信扫码关注我们:跟着数理化走天下


获得更多的信息哦,一起交流,一起成长哦:微信号:跟着数理化走天下,纯属个人的交流,无盈利目的


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值