场景
实现了车辆的轮廓识别,并且已经提取轮廓的最小矩形范围,现在需要知道车尾离矩形最近的两个点,可能有点大材小用
参考
http://www.cnblogs.com/mikewolf2002/p/3427564.html
http://blog.csdn.net/zhuoyueljl/article/details/53588271
代码
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include
using namespace cv;
using namespace std;
void Dilate( InputArray src, OutputArray dst)
{
int dilation_type = MORPH_RECT;
int dilation_size = 10;
Mat dielem = getStructuringElement( dilation_type,
Size( 2*dilation_size + 1, 2*dilation_size+1 ),
Point( dilation_size, dilation_size ) );
///膨胀操作
dilate( src, dst, dielem );
}
/*
该函数主要是捕获图片中完整出现轮廓的车辆,判断条件为
车辆的轮廓的Y坐标不能大于图片的长度,其次过滤掉面积过小
的轮廓,很可能是车镜或者帧间差分将车辆拆分成两段的误差
*/
void CaptureCompleteVehicle(Mat &srcMat, Mat &grayMat)
{
vector
vector
findContours(grayMat, contours,hierarchy,RETR_EXTERNAL,CHAIN_APPROX_NONE,Point());
Mat dstMat=Mat::zeros(grayMat.size(),CV_8UC1);
Mat contourMat;
srcMat.copyTo(contourMat);
int picHeight = grayMat.size().height;
bool bTouchBotton = false;
bool bTouchTop = false;
vector
for (int i=0; i { mu[i] = moments(contours[i], false); } vector for (int i=0; i { mc[i] = Point2d(mu[i].m10 / mu[i].m00, mu[i].m01 / mu[i].m00); } for(int i=0; i { if (contourArea(contours[i]) < 10000) continue; bTouchBotton = false; bTouchTop = false; for (int k=0; k { Point2f pos = contours[i][k]; if ((pos.y +10) > picHeight) { bTouchBotton = true; break; } if (pos.y == 0) { bTouchTop = true; break; } } if (bTouchBotton || bTouchTop) continue; drawContours(dstMat, contours, i, Scalar(255, 0, 0), 1, 8, hierarchy); RotatedRect rect=minAreaRect(contours[i]); Point2f P[4]; rect.points(P); int leftBottonIndex = 0; for(int j=0; j<=3; j++) { line(contourMat, P[j], P[(j+1)%4], Scalar(255, 0, 0), 2); if ((P[j].x < mc[i].x) && (P[j].y > mc[i].y)) { leftBottonIndex = j; } } cv::Rect re(P[leftBottonIndex].x - 20, P[leftBottonIndex].y - 20 , 40, 40); rectangle(contourMat, re, Scalar(0, 255, 0), 4); circle(contourMat, mc[i], 5, Scalar(0, 0, 255), -1, 8, 0); } imshow("NewAreaRect", contourMat); } int main(int argc,char *argv[]) { VideoCapture videoCap("E:/smoky-cars/positive/大庆东路与水机路交叉口(东北)_冀BU0157_02_141502_01_3_50.wh264"); if(!videoCap.isOpened()) return -1; double videoFPS=videoCap.get(CV_CAP_PROP_FPS); //获取帧率 double videoPause=1000/videoFPS; Mat framePrePre; //上上一帧 Mat framePre; //上一帧 Mat frameNow; //当前帧 Mat frameDet; //运动物体 videoCap>>framePrePre; videoCap>>framePre; cvtColor(framePrePre,framePrePre,CV_RGB2GRAY); cvtColor(framePre,framePre,CV_RGB2GRAY); int save=0; while(true) { videoCap>>frameNow; if(frameNow.empty()||waitKey(videoPause)==27) break; cvtColor(frameNow,frameNow,CV_RGB2GRAY); Mat Det1; Mat Det2; absdiff(framePrePre,framePre,Det1); //帧差1 absdiff(framePre,frameNow,Det2); //帧差2 threshold(Det1,Det1,0,255,CV_THRESH_OTSU); //自适应阈值化 threshold(Det2,Det2,0,255,CV_THRESH_OTSU); Mat element=getStructuringElement(0,Size(3,3)); //膨胀核 dilate(Det1,Det1,element); //膨胀 dilate(Det2,Det2,element); bitwise_and(Det1,Det2,frameDet); framePrePre=framePre; framePre=frameNow; Dilate(frameDet, frameDet); CaptureCompleteVehicle(frameNow, frameDet); waitKey(1000); } return 0; }