简单车道线检测算法

github的代码,添加了一点自己的改动和注释
大概思想 滤波 增强 roi区域划分 canny边缘提取 hough直线检测
主要的超参在cannny检测阈值以及hough直线检测阈值上
roadDetection.h

#pragma once

#include <string>
#include<vector>

#include <opencv.hpp>
#include<opencv2/core.hpp>
#include<opencv2/highgui.hpp>
#include<opencv2/imgproc.hpp>
#include "opencv2/imgproc/imgproc_c.h"
//B样条曲线拟合
//双曲线拟合
using namespace std;
using namespace cv;

#define INPUT "road\\001.png"
#define OUTPUT "result\\001.png"

class RoadDetection
{
public:
	RoadDetection(const Mat& image=Mat ())
		:readImage(image)
	{
		left_flag = false;
		right_flag = false;
		writeImage = readImage;
	}

	Mat result()
	{
		return writeImage;
	}

	//去噪
	void deNoise(Mat inputImage, Mat& outputImage);
	//设置ROI
	void mask(Mat inputImage, Mat& outputImage);
	//形态学闭运算
	void CloseOperation(Mat inputImage, Mat& outputImage);
	//亮度特征
	void Brightness(Mat inputImage, Mat& outputImage);
	//颜色阈值
	void colorThreshold(Mat inputImage, Mat& outputImage);
	void HistEqualize(Mat inputImage, Mat& outputImage);
	//边缘检测
	void edgeDetector(Mat inputImage, Mat& outputImage);
	//霍夫线概率变换
	vector<Vec4i> houghLines(Mat inputImage);
	//筛选线段
	void lineSeparation(const vector<Vec4i>& lines, vector<std::vector<cv::Vec4i> >& output);
	void select_lines(const vector<std::vector<cv::Vec4i> >& output, vector<std::vector<cv::Vec4i> >& selectLines);
	vector<Point> regression(vector<vector<Vec4i> > left_right_lines, Mat inputImage);

//	void findSide(const Mat& inputImage);
	void findTop(vector<vector<Vec4i> > output);

	bool predictTurn(int right_x, int left_x);
	bool polynomial_curve_fit(vector<Vec4i> lines, int n);
	//检测车道线主流程
	void detection();

private:
	Mat readImage;
	Mat writeImage;
	bool left_flag;
	bool right_flag;
	Point right_b;	 //右直线点
	double right_m;  //右直线斜率
	Point left_b;	 //左直线点
	double left_m;   //左直线斜率
	int top_y;
	int canTh = 150;
};

roadDetection.cpp

#include<string>
#include<vector>
#include<iostream>
#include"roadDetection.h"
#include<time.h>

//#define __TEST__
#define __DEBUG__

using namespace std;
using namespace cv;

//调试时显示处理后的图片
void myShow(std::string str, Mat inputImage)
{
#ifdef __DEBUG__
	imshow(str, inputImage);
#endif
}

void testShow(std::string str, Mat inputImage)
{
#ifdef __TEST__
	imshow(str, inputImage);
#endif
}

//高斯去噪
void RoadDetection::deNoise(Mat inputImage, Mat& outputImage)
{
	GaussianBlur(inputImage, outputImage, Size(3, 3), 0, 0);
	//blur(inputImage, outputImage, Size(7, 7));	//均值
	//medianBlur(inputImage, outputImage, 7);			//中值
	//bilateralFilter(inputImage, outputImage, 25, 25 * 2, 25 / 2);    //双边
}

//闭运算
void RoadDetection::CloseOperation(Mat inputImage, Mat& outputImage)
{
	Mat element = getStructuringElement(MORPH_RECT, Size(7, 7));
	//进行形态学操作
	//morphologyEx(inputImage, outputImage, MORPH_GRADIENT, element);

	//闭运算
	morphologyEx(inputImage, outputImage, MORPH_CLOSE, element);

	//morphologyEx(inputImage, outputImage, MORPH_OPEN, element);
	//morphologyEx(inputImage, outputImage, MORPH_DILATE, element);

	//膨胀
	//dilate(inputImage, outputImage, element);
	//腐蚀
	//erode(inputImage, outputImage, element);
}

//设置ROI
void RoadDetection::mask(Mat inputImage, Mat& outputImage) {
	Mat mask = Mat::zeros(inputImage.size(), inputImage.type());
	Point pts[4] = {
		//Point(inputImage.cols / 4, inputImage.rows / 4),
		Point(0, inputImage.rows>>1),
		//Point(inputImage.cols /4, inputImage.rows >> 1),
		Point(0, inputImage.rows),
		Point(inputImage.cols, inputImage.rows),
		Point(inputImage.cols, inputImage.rows >>1)
		//Point(inputImage.cols/4*3, inputImage.rows >> 1)
	};

	// Create a binary polygon mask
	fillConvexPoly(mask, pts, 4, Scalar(255, 255, 255));
	// Multiply the edges image and the mask to get the output
	bitwise_and(inputImage, mask, outputImage);

	定义一个Mat类型并给其设定ROI区域
	//Mat imageROI;
	方法一
	//imageROI = inputImage(Rect(0, inputImage.rows >> 1, inputImage.cols, inputImage.rows>>1));
}

Mat Histogram(Mat inputImage)
{
	/// 分割成3个单通道图像 ( R, G 和 B )
	vector<Mat> rgb_planes;
	split(inputImage, rgb_planes);

	/// 设定bin数目
	int histSize = 255;

	/// 设定取值范围 ( R,G,B) )
	float range[] = { 0, 255 };
	const float* histRange = { range };

	bool uniform = true; bool accumulate = false;

	Mat r_hist, g_hist, b_hist;

	/// 计算直方图:
	calcHist(&rgb_planes[0], 1, 0, Mat(), r_hist, 1, &histSize, &histRange, uniform, accumulate);
	//calcHist(&rgb_planes[1], 1, 0, Mat(), g_hist, 1, &histSize, &histRange, uniform, accumulate);
	//calcHist(&rgb_planes[2], 1, 0, Mat(), b_hist, 1, &histSize, &histRange, uniform, accumulate);

	// 创建直方图画布
	int hist_w = 400; int hist_h = 400;
	int bin_w = cvRound((double)hist_w / histSize);

	Mat histImage(hist_w, hist_h, CV_8UC3, Scalar(0, 0, 0));

	/// 将直方图归一化到范围 [ 0, histImage.rows ]
	normalize(r_hist, r_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());
	//normalize(g_hist, g_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());
	//normalize(b_hist, b_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());

	/// 在直方图画布上画出直方图
	for (int i = 1; i < histSize; i++)
	{
		line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(r_hist.at<float>(i - 1))),
			Point(bin_w*(i), hist_h - cvRound(r_hist.at<float>(i))),
			Scalar(255, 255, 255), 2, 8, 0);
		//line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(g_hist.at<float>(i - 1))),
		//	Point(bin_w*(i), hist_h - cvRound(g_hist.at<float>(i))),
		//	Scalar(0, 255, 0), 2, 8, 0);
		//line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(b_hist.at<float>(i - 1))),
		//	Point(bin_w*(i), hist_h - cvRound(b_hist.at<float>(i))),
		//	Scalar(255, 0, 0), 2, 8, 0);
	}
	return histImage;
}

//直方图均衡化
void RoadDetection::HistEqualize(Mat inputImage, Mat& outputImage)
{
	cvtColor(inputImage, inputImage, CV_RGB2GRAY);
	testShow("cvtColor", inputImage);

	testShow("均衡化前直方图", Histogram(inputImage));
	vector<Mat> splitBGR(inputImage.channels());
	split(inputImage, splitBGR);

	for (int i = 0; i < inputImage.channels(); ++i)
	{
		equalizeHist(splitBGR[i], splitBGR[i]);
	}
	Mat mergeImg;
	merge(splitBGR, mergeImg);
	mergeImg.copyTo(outputImage);

	testShow("均衡化后直方图", Histogram(outputImage));

	//outputImage.convertTo(outputImage, -1, 1, 20);
	//testShow("convertTo", outputImage);
}

//亮度特征
void RoadDetection::Brightness(Mat inputImage, Mat& outputImage)
{
	//cvtColor(inputImage, inputImage, CV_RGB2GRAY);
	//testShow("cvtColor", inputImage);
	
	//Mat tmp1,tmp2;
	//threshold(inputImage, tmp1, 140, 255, THRESH_TOZERO);
	//threshold(inputImage, tmp1, 245, 255, THRESH_BINARY);
	//testShow("250", tmp1);
	//threshold(inputImage, tmp2, 200, 255, THRESH_BINARY);
	//testShow("200", tmp2);
	//bitwise_and(tmp1, tmp2, outputImage);
	//addWeighted(tmp1, 1, tmp2, 1, 0.0, outputImage);

	threshold(inputImage, outputImage, 100, 255, THRESH_BINARY);
}

//颜色阈值
void RoadDetection::colorThreshold(Mat inputImage, Mat& outputImage)
{	
	//cvtColor(inputImage, outputImage, CV_RGB2GRAY);
	//threshold(inputImage, outputImage, 140, 255, THRESH_BINARY);
	Mat bgr,hsv, mask,result;
	testShow("input", inputImage);
	inputImage.convertTo(bgr, CV_32FC3, 1.0 / 255, 0);
	testShow("bgr", bgr);
	cvtColor(bgr, hsv, COLOR_BGR2HSV);
	testShow("COLOR_BGR2HSV", hsv);
	
	inRange(hsv, Scalar(0, 0, (float)(230)/255), Scalar(90, (float)255/255, (float)(255)/255), mask);
	//inRange(hsv, Scalar(0, 0, (float)(150) / 255), Scalar(90, (float)255 / 255, (float)(255) / 255), mask);
	testShow("inRange", mask);

	//只保留
	//result = Mat::zeros(bgr.size(), CV_32FC3);
	//for (int r = 0; r < bgr.rows; r++)
	//{
	//	for (int c = 0; c < bgr.cols; c++)
	//	{
	//		if (mask.at<uchar>(r, c) == 255)
	//		{
	//			result.at<Vec3f>(r, c) = bgr.at<Vec3f>(r, c);
	//		}
	//	}
	//}
	result = mask;
	result.convertTo(result, CV_8UC3, 255.0, 0);
	outputImage = result.clone();
}

//边缘检测
void RoadDetection::edgeDetector(Mat inputImage, Mat& outputImage)
{
	//testShow("threshold", outputImage);
	cout << canTh << endl;
	Canny(inputImage, outputImage, canTh, 10, 3);  //Canny边缘检测算子
	//imshow("111111", outputImage);
	//waitKey(0);
	//Laplacian(inputImage, outputImage, CV_16S, 3, 1, 0, BORDER_DEFAULT); //拉普拉斯算子
	//convertScaleAbs(outputImage, outputImage);

	//Soble边缘检测算子
	//Mat grad_x, grad_y;
	//Mat abs_grad_x, abs_grad_y;

	【3】求 X方向梯度
	//Sobel(inputImage, grad_x, CV_16S, 1, 0, 3, 1, 1, BORDER_DEFAULT);
	//convertScaleAbs(grad_x, abs_grad_x);
	//imshow("【效果图】 X方向Sobel", abs_grad_x);

	【4】求Y方向梯度
	//Sobel(inputImage, grad_y, CV_16S, 0, 1, 3, 1, 1, BORDER_DEFAULT);
	//convertScaleAbs(grad_y, abs_grad_y);
	//imshow("【效果图】Y方向Sobel", abs_grad_y);

	【5】合并梯度(近似)
	//addWeighted(abs_grad_x, 0.5, abs_grad_y, 0.5, 0, outputImage);
	//imshow("【效果图】整体方向Sobel", outputImage);
}


//霍夫线概率变换
vector<Vec4i> RoadDetection::houghLines(Mat inputImage)
{
	vector<Vec4i> line;

	HoughLinesP(inputImage, line, 1, CV_PI / 180, 40, 20, 10);

	return line;
}

//画线
void drawDetectedLines(Mat &image, vector<Vec4i> lines, Scalar color = Scalar(255, 255, 255))
{
	vector<Vec4i>::const_iterator it2 = lines.begin();
	while (it2 != lines.end()) {
		Point pt1((*it2)[0], (*it2)[1]);
		Point pt2((*it2)[2], (*it2)[3]);
		line(image, pt1, pt2, color);
		++it2;
	}
	
}

//selected_lines
void RoadDetection::lineSeparation(const vector<Vec4i>& lines, vector<std::vector<cv::Vec4i> >& output)
{
	size_t j = 0;
	Point ini;
	Point fini;
	double slope_thresh = 0.3;
	vector<double> slopes;
	vector<Vec4i> selected_lines;
	vector<Vec4i> right_lines, left_lines;

	// 根据斜率筛选
	for (auto i : lines) 
	{
		ini = cv::Point(i[0], i[1]);
		fini = cv::Point(i[2], i[3]);

		// Basic algebra: slope = (y1 - y0)/(x1 - x0)
		//计算斜率
		double slope = (static_cast<double>(fini.y) - static_cast<double>(ini.y)) / (static_cast<double>(fini.x) - static_cast<double>(ini.x) + 0.00001);

		// If the slope is too horizontal, discard the line
		// If not, save them  and their respective slope
		//如果这条线的斜率太过于水平(斜率大于0.3),忽略这条直线
		//否则保存同时保存斜率
		if (std::abs(slope) > slope_thresh) 
		{
			slopes.push_back(slope);
			selected_lines.push_back(i);
		}
	}
	// selected_lines存储了所有斜率满足要求的直线的两个点Vec4i,slopes存储了他们对应的斜率
	// Split the lines into right and left lines
	//依据图像的中心点来区分左右 列数/2
	double img_center = static_cast<double>((readImage.cols / 2));
	while (j < selected_lines.size()) {
		//循环遍历所有筛选的直线 ini为起始点 fini为结束点
		ini = Point(selected_lines[j][0], selected_lines[j][1]);
		fini = Point(selected_lines[j][2], selected_lines[j][3]);

		// Condition to classify line as left side or right side
		//将筛选的直线依据刚才求得中心分为左边、右边 
		//并且在右边的线(起点、终点的x值大于中心点)的斜率应该大于0且小于0.5
		//并且在左边的线(起点、终点的x值小于中心点)的斜率应该小于0
		if ((slopes[j] > 0.3 && slopes[j] < 0.5) && fini.x > img_center && ini.x > img_center) {
			right_lines.push_back(selected_lines[j]);
			right_flag = true;
		}
		else if ((slopes[j] < -0.3 && slopes[j] > -0.40 )&& fini.x < img_center && ini.x < img_center) {
			left_lines.push_back(selected_lines[j]);
			left_flag = true;
		}
		j++;
	}

	output[0] = right_lines;
	output[1] = left_lines;

}

void RoadDetection::select_lines(const vector<std::vector<cv::Vec4i> >& output, vector<std::vector<cv::Vec4i> >& selectLines)
{
	for (auto i : output[1])
	{
		cout << i << endl;
		
	}
}

vector<Point> RoadDetection::regression(vector<vector<Vec4i> > left_right_lines, Mat inputImage)
{
	vector<Point> output(4);
	Point ini;
	Point fini;
	Point ini2;
	Point fini2;
	Vec4d right_line;
	Vec4d left_line;
	vector<Point> right_pts;
	vector<Point> left_pts;

	// If right lines are being detected, fit a line using all the init and final points of the lines
	//如果检测到存在右边的线,就获取一条直线来替代所有的直线
	if (right_flag == true) {
		//将右边区域所有线段的端点(起点和终点)加入到right_pts中,利用fitLine来求解一条直线
		// 所求直线思想:求取一条到right_pts中所有点距离最近的直线,可采用L2距离及欧氏距离
		//line[0]、line[1] 存放的是直线的方向向量,
		//double cos_theta = line[0]; double sin_theta = line[1];
		//line[2]、line[3] 存放的是直线上一个点的坐标。
		for (auto i : left_right_lines[0]) {
			ini = Point(i[0], i[1]);
			fini = Point(i[2], i[3]);

			right_pts.push_back(ini);
			right_pts.push_back(fini);
		}

		if (right_pts.size() > 0) {
			// The right line is formed here
			fitLine(right_pts, right_line, DIST_L2, 0, 0.01, 0.01);
			right_m = right_line[1] / right_line[0];
			right_b = Point(right_line[2], right_line[3]);
		}
	}
	//同右边的处理
	// If left lines are being detected, fit a line using all the init and final points of the lines
	if (left_flag == true) {
		for (auto j : left_right_lines[1]) {
			ini2 = cv::Point(j[0], j[1]);
			fini2 = cv::Point(j[2], j[3]);

			left_pts.push_back(ini2);
			left_pts.push_back(fini2);
		}

		if (left_pts.size() > 0) {
			// The left line is formed here
			fitLine(left_pts, left_line, DIST_L2, 0, 0.01, 0.01);
			left_m = left_line[1] / left_line[0];
			left_b = Point(left_line[2], left_line[3]);
		}
	}

	// One the slope and offset points have been obtained, apply the line equation to obtain the line points
	//依据上文求的斜率left_m和直线上的一个点的left_b求解出直线
	//求当y为 ini_y  fin_y时的x,求得左右两条车道线的起点和终点
	cout << "右边斜率:" << right_m << endl;
	cout << "左边斜率:" << left_m << endl;
	int ini_y = inputImage.rows;
	int fin_y = inputImage.rows/3*2;

	double right_ini_x = ((ini_y - right_b.y) / right_m) + right_b.x;
	double right_fin_x = ((fin_y - right_b.y) / right_m) + right_b.x;

	double left_ini_x = ((ini_y - left_b.y) / left_m) + left_b.x;
	double left_fin_x = ((fin_y - left_b.y) / left_m) + left_b.x;

	output[0] = Point(right_ini_x, ini_y);
	output[1] = Point(right_fin_x, fin_y);
	output[2] = Point(left_ini_x, ini_y);
	output[3] = Point(left_fin_x, fin_y);

	return output;
}

bool RoadDetection::predictTurn(int right_x, int left_x) {
	std::string output;
	double vanish_x;
	double thr_vp = 32;
	double img_center = (right_x + left_x)>>1;    //车道中心点
	// The vanishing point is the point where both lane boundary lines intersect
	vanish_x = static_cast<double>(((right_m*right_b.x) - (left_m*left_b.x) - right_b.y + left_b.y) / (right_m - left_m));

	// The vanishing points location determines where is the road turning
	//if (vanish_x >= (img_center - thr_vp) && vanish_x <= (img_center + thr_vp))
	//	output = "Straight";
	//else if (vanish_x < (img_center - thr_vp))
	//	output = "Left Turn";
	//else if (vanish_x > (img_center + thr_vp))
	//	output = "Right Turn";
	if (vanish_x >= (img_center - thr_vp) && vanish_x <= (img_center + thr_vp))
		return true;
	else
		return false;

	/*circle(writeImage, Point(img_center, writeImage.rows / 3 * 2), 5, Scalar(0, 0, 0), 2, 8, 0);
	circle(writeImage, Point(vanish_x, writeImage.rows>>1), 5, Scalar(0, 0, 0), 2, 8, 0);*/

	//cout << "right_b " << right_b << " right_m " << right_m <<" left_b "<< left_b <<" left_m " << left_m << endl;
	//cout << vanish_x - img_center << endl;
	//return output;
}


//bool RoadDetection::polynomial_curve_fit(vector<Vec4i> lines,  int n, bool left_flge)
//{
//	vector<Point2f> key_point;
//	vector<Point2f> poly_point;
//	Mat A;
//	int min_x=10000, max_x=0;
//	for (auto i : lines)
//	{
//		key_point.push_back(Point2f(i[0], i[1]));
//		key_point.push_back(Point2f(i[2], i[3]));
//		if (max_x < (i[0]>i[2]?i[0]:i[2]))
//			max_x = (i[0] > i[2] ? i[0] : i[2]);
//		if (min_x > (i[0] < i[2] ? i[0] : i[2]))
//			min_x = (i[0] < i[2] ? i[0] : i[2]);
//	}
//
//	if (left_flge)
//	{
//		min_x = left_x;
//	}
//	else
//	{
//		max_x = right_x;
//	}
//	//Number of key points
//	int N = key_point.size();
//	//构造X
//	Mat X = Mat::zeros(n + 1, n + 1, CV_64FC1);
//	for (int i = 0; i < n + 1; i++)
//	{
//		for (int j = 0; j < n + 1; j++)
//		{
//			for (int k = 0; k < N; k++)
//			{
//				X.at<double>(i, j) = X.at<double>(i, j) + pow(key_point[k].x, i + j);
//			}
//		}
//	}
//	//构造Y
//	Mat Y = Mat::zeros(n + 1, 1, CV_64FC1);
//	for (int i = 0; i < n + 1; i++)
//	{
//		for (int k = 0; k < N; k++)
//		{
//			Y.at<double>(i, 0) = Y.at<double>(i, 0) + pow(key_point[k].x, i) * key_point[k].y;
//		}
//	}
//
//	A = Mat::zeros(n + 1, 1, CV_64FC1);
//	//求解A
//	solve(X, Y, A, DECOMP_LU);
//	for (float x = min_x; x < max_x; x++)
//	{
//		double y = A.at<double>(0, 0) + A.at<double>(1, 0) * x +
//			A.at<double>(2, 0)*pow(x, 2) + A.at<double>(3, 0)*pow(x, 3);
//
//		poly_point.push_back(Point2f(x, y));
//	}
//
//	Point2f now, next;
//	vector<Point2f>::iterator it = poly_point.begin();
//	next = *it++;
//	while (it != poly_point.end())
//	{
//		now = next;
//		next = *it++;
//		line(writeImage, now, next, Scalar(255, 255, 0), 5, 16);
//	}
//	
//	cout << A << endl;
//	cout << poly_point << endl;
//	return true;
//}

bool RoadDetection::polynomial_curve_fit(vector<Vec4i> lines, int n)
{
	vector<Point2f> key_point;
	vector<Point2f> poly_point;
	Mat A;

	for (auto i : lines)
	{
		key_point.push_back(Point2f(i[1], i[0]));
		key_point.push_back(Point2f(i[3], i[2]));
	}

	//Number of key points
	int N = key_point.size();
	//构造X
	Mat X = Mat::zeros(n + 1, n + 1, CV_64FC1);
	for (int i = 0; i < n + 1; i++)
	{
		for (int j = 0; j < n + 1; j++)
		{
			for (int k = 0; k < N; k++)
			{
				X.at<double>(i, j) = X.at<double>(i, j) + pow(key_point[k].x, i + j);
			}
		}
	}
	//构造Y
	Mat Y = Mat::zeros(n + 1, 1, CV_64FC1);
	for (int i = 0; i < n + 1; i++)
	{
		for (int k = 0; k < N; k++)
		{
			Y.at<double>(i, 0) = Y.at<double>(i, 0) + pow(key_point[k].x, i) * key_point[k].y;
		}
	}

	A = Mat::zeros(n + 1, 1, CV_64FC1);
	//求解A
	solve(X, Y, A, DECOMP_LU);
	for (float x = top_y; x < writeImage.rows; x++)
	{
		double y = A.at<double>(0, 0) + A.at<double>(1, 0) * x +
			A.at<double>(2, 0)*pow(x, 2) + A.at<double>(3, 0)*pow(x, 3);

		poly_point.push_back(Point2f(y, x));
	}

	Point2f now, next;
	vector<Point2f>::iterator it = poly_point.begin();
	next = *it++;
	while (it != poly_point.end())
	{
		now = next;
		next = *it++;
		line(writeImage, now, next, Scalar(255, 255, 0), 5, 16);
	}

//	cout << A << endl;
//	cout << poly_point << endl;
	return true;
}

//void RoadDetection::findSide(const Mat& inputImage)
//{
//	bool left = true;
//	//cout << inputImage.cols << " " << inputImage.rows << endl; //690 456
//	for (int x = 0; x < inputImage.cols; ++x)
//	{
//		for (int y = inputImage.rows-1; y > inputImage.rows-10; --y)
//		{
//			if (left&&inputImage.at< uchar>(y, x) == 255)
//			{
//				left = false;
//				left_x = x;
//			}
//			if (inputImage.at<uchar>(y, x) == 255)
//			{
//				right_x = x;
//			}
//			//cout << (int)inputImage.at<uchar>(x, y) << " ";
//		}
//		cout <<x<<" ";
//	}
//	cout << left_x << " " << right_x << endl;
//}
void RoadDetection::findTop(vector<vector<Vec4i> > output)
{
	top_y = writeImage.rows;
	for (auto i : output[0])
	{
		if (top_y > (i[1] < i[3] ? i[1] : i[3]))
		{
			top_y = (i[1] < i[3] ? i[1] : i[3]);
		}
	}
	for (auto i : output[1])
	{
		if (top_y > (i[1] < i[3] ? i[1] : i[3]))
		{
			top_y = (i[1] < i[3] ? i[1] : i[3]);
		}
	}
}

void RoadDetection::detection()
{
	vector<Vec4i> lines;
	vector<vector<Vec4i> > output(2),selectLines(2);
	vector<Point> lane;
	Mat midImage,tmpImage= readImage.clone();
	Mat drawImage(readImage.size(), readImage.type(), Scalar(0,0,0));
	Mat brightImage, colorImage, edgeImg;
	//myShow("【原图】", readImage);

	deNoise(readImage, midImage);
	//myShow("【滤波】", midImage);

	//CloseOperation(midImage, midImage);
	//myShow("【闭运算】", midImage);

	HistEqualize(midImage, brightImage);
	//myShow("【直方图均衡化】", brightImage);

	mask(midImage, midImage);
	//myShow("【ROImidImage】", midImage);
	mask(brightImage, brightImage);
	//myShow("【ROI】", brightImage);

	//Brightness(brightImage, brightImage);
	//myShow("【亮度提取】", brightImage);

	//colorThreshold(midImage, colorImage);
	//myShow("【颜色阈值】", colorImage);

	//addWeighted(brightImage, 1, colorImage, 1, 0.0, midImage);
	//myShow("【特征提取】", midImage);


	edgeDetector(midImage, edgeImg);
	//myShow("【边缘检测】", midImage);

	lines = houghLines(edgeImg);

	//drawDetectedLines(tmpImage, lines, Scalar(0, 255, 0));
	//myShow("lines", tmpImage);

	lineSeparation(lines, output);
	//drawDetectedLines(writeImage, output[0], Scalar(0, 255, 0));
	//drawDetectedLines(writeImage, output[1], Scalar(0, 0, 255));
	// 
	

	if (left_flag == 0 || right_flag == 0) {
		if (canTh == 150) {
			cout << "再次检测" << endl;
			//表示检测未成功 降低cannny的阈值再试一下
			canTh = 75;
			edgeDetector(midImage, midImage);
			//myShow("【边缘检测】", midImage);

			lines = houghLines(midImage);

			//drawDetectedLines(tmpImage, lines, Scalar(0, 255, 0));
			//myShow("lines", tmpImage);

			lineSeparation(lines, output);
			drawDetectedLines(writeImage, output[0], Scalar(0, 255, 0));
			drawDetectedLines(writeImage, output[1], Scalar(0, 0, 255));
			myShow("final lines", writeImage);
		}
		
	}

	if (left_flag == 0 || right_flag == 0) {
		cout << "检测失败" << endl;
		return;
	}

	//select_lines(output, selectLines);
	//output[0]右边所有线的集合 output[1]左边所有线的集合 存的都是Vec4i
	drawDetectedLines(tmpImage, output[0], Scalar(0, 255, 0));
	drawDetectedLines(tmpImage, output[1], Scalar(0, 0, 255));
	//myShow("lines", tmpImage);


	//drawDetectedLines(tmpImage, lines, Scalar(0, 255, 0));


	lane = regression(output, midImage);

	/*if (predictTurn(lane[0].x, lane[2].x))
	{
		line(writeImage, lane[0], lane[1], Scalar(0, 255, 255), 5, 16);
		line(writeImage, lane[2], lane[3], Scalar(0, 255, 255), 5, 16);
	}
	else
	{
		findTop(output);
		polynomial_curve_fit(output[0],3);
		polynomial_curve_fit(output[1], 3);
	}*/
	line(writeImage, lane[0], lane[1], Scalar(0, 255, 255), 5, 16);
	line(writeImage, lane[2], lane[3], Scalar(0, 255, 255), 5, 16);
}

main.cpp

#include <io.h>
#include <iostream>
#include<string>
#include <vector> 
#include <opencv.hpp>
#include "roadDetection.h"

using namespace std;
using namespace cv;


获取特定格式的文件名    
//void getAllFiles(string path, vector<string>& files, string format)
//{
//	long  hFile = 0;//文件句柄  64位下long 改为 intptr_t
//	struct _finddata_t fileinfo;//文件信息 
//	string p;
//	if ((hFile = _findfirst(p.assign(path).append("\\*" + format).c_str(), &fileinfo)) != -1) //文件存在
//	{
//		do
//		{
//			if ((fileinfo.attrib & _A_SUBDIR))//判断是否为文件夹
//			{
//				if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0)//文件夹名中不含"."和".."
//				{
//					files.push_back(p.assign(path).append("\\").append(fileinfo.name)); //保存文件夹名
//					getAllFiles(p.assign(path).append("\\").append(fileinfo.name), files, format); //递归遍历文件夹
//				}
//			}
//			else
//			{
//				files.push_back(p.assign(path).append("\\").append(fileinfo.name));//如果不是文件夹,储存文件名
//			}
//		} while (_findnext(hFile, &fileinfo) == 0);
//		_findclose(hFile);
//	}
//}


//int main()
//{
//	//vector<string> files;
//	//string filePath = "D:\\用户文件夹\\Documents\\Visual Studio 2017\\Projects\\道路检测\\道路检测\\road\\";
//	//string format = ".jpg";
//	//getAllFiles(filePath, files, format);
//	//for (int i = 0; i < files.size(); i++)
//	//{
//		Mat readimage;
//		string readStr = INPUT;
//		string writeStr = OUTPUT;
//
//		readimage = imread(readStr);
//		if (!readimage.data) { printf("读取Image错误~! \n"); return -1; }
//		imshow("原图", readimage);
//
//		RoadDetection RoDetec(readimage);
//		RoDetec.detection();
//
//		Mat writeimage = RoDetec.result();
//		imshow("检测图", writeimage);
//		imwrite(writeStr, writeimage);
//
//		waitKey(0);
	}
//	
//	return 0;
//}
  
void loop(string readStr, string writeStr)
{
	readStr = readStr;
	writeStr = "result\\" + writeStr;
	Mat readimage;
	readimage = imread(readStr);
	if (!readimage.data) { printf("读取Image错误~! \n"); return;}
	//imshow(readStr, readimage);

	RoadDetection RoDetec(readimage);
	RoDetec.detection();

	Mat writeimage = RoDetec.result();
	//imwrite(writeStr, writeimage);

	imshow(writeStr, writeimage);
}


int main()
{
	string folder_path = "repos/test_lane/data/*.png";
	vector<cv::String> file_names;
	glob(folder_path, file_names);
	cv::Mat img;
	//cout << file_names.size() << endl;
	
	for (int i = 0; i < file_names.size(); i++) {
		clock_t s = clock();
		loop(file_names[i], file_names[i]);
		clock_t e = clock();
		cout << double(e - s) << "ms" << endl;
		waitKey(0);
	}
	
	//string str1 = "./data/7.png";
	//loop(str1, str1);
	//waitKey(0);
}