自己动手写一个霍夫变换检测直线(线段)

霍夫变换最简单的使用就是检测直线了。
原理:
对于x-y坐标中的一条直线y=kx+b,任取其上两点(x1,y1)与(x2,y2),在参数坐标k-b中分别对应两条直线L1与L2,两者相交与一点(b,k)。原直线上所有点在参数平面中对应的直线都相交于这一点。
对于x=c这类直线无法用y=kx+b处理,一般使用参数方程rho=xcos(theta)+ysin(theta)表示,则参数平面中为众多三角函数曲线相交与一点(theta,rho)
对于参数平面细分(因为程序处理时候数据不会连续,是离散的),分别统计每个点挂载的曲线数量,即“有多少条直线在这一点相交”,取数量最多的那个点,对应原有的待检测直线
先用霍夫变换检测直线,确定“伪直线”,然后取原图中点,用最小二乘法求出“真直线”。

说明:
OpenCV中自带HoughLines和HoughLinesP等函数用于检测直线。
代码

/*读入一张图像
*检测其中的一条线段
*并将其标记出来
*策略:取图像上每一点做霍夫变换,然后统计交点
*用交点的信息推出直线的斜率与截距
*并用点线距离判别合法点
*再用最小二乘法模拟直线方程
***ChriZ!
*/
#include <cv.h>
#include <highgui.h>
#include <iostream>
#include <string>
#include <math.h>
#include <vector>
#include <string.h>
#include <map>

using namespace std;
using namespace cv;

//0 represent black
//1 represent white
#define LINE 0
#define ELSE 255
typedef Mat_<uchar> MatU;

double f(int xi, int yi, double theta){
    double ret=xi*cos(theta)+yi*sin(theta);
    return ret;
}

double dis(double k, double b, double xi, double yi){
    double up=fabs(k*xi-yi+b);
    double down=sqrt(k*k+1);
    return up/down;
}

int main(){
    string filename="C:/testdir/13.png";
    Mat imgg=imread(filename);

    MatU img;
    cvtColor(imgg, img, CV_RGB2GRAY);
   // Mat imhough=Mat(img.rows, img.cols, img.type());

  //  bitwise_not(img, img);
    //threshold(img, img, 128, 255, THRESH_BINARY);
    int x, y;
    double theta;
    //MatU test=MatU(img.rows, img.cols, img.type());
    //Matrix test=new Matrix(img.rows, img.cols);
    //int test[img.rows*4][img.cols];
    int test[314][800];
    for(x=0; x<314; x++){
        for(y=0; y<800; y++){
            test[x][y]=0;
        }
    }

    for(y=0; y<img.rows; y++){
        for(x=0; x<img.cols; x++){
            Point cur(x,y);
            if(img(cur)==LINE){
             //   cout << "(" << i << "," << j << ")" << endl;
                for(theta=0; theta<3.14; theta=theta+0.01){
                    double rho=f(x, y, theta);
                  //  if(rho>150)   cout << rho << endl;
                    test[int(theta*100)][int(rho)+400]+=1;
                  //  数组下标非负,遭遇负数rho时要处理,所以统统加400,后面处理时再减400
                  //  test[theta][int(rho)]+=1;
                }
            }
        }
    }

    int max=-1;
    double mx=0, my=0;
    for(y=0; y<800; y++){
        for(x=0; x<314; x++){
            if(test[x][y]>max){
                max=test[x][y];
                mx=x/100.0;
                my=y-400;
            }
        }
    }
    //cout << "max=" << max << endl;
    //cout << "mx=" << mx << endl;
    double mk=-1.0/tan(mx);
    double mb=my*1.0/sin(mx);
    cout << "k=" << mk << endl << "b=" << mb << endl;

   
    double dk=1;//pivot valut

    vector<Point> store;
    double xmin=200, xmax=0;//[xmin,xmax]表示直线横坐标范围
    for(y=0; y<img.rows; y++){
        for(x=0; x<img.cols; x++){
            Point cur(x, y);
            if(img(cur)==LINE){
                if(dis(mk, mb, x, y)<dk){
                    store.push_back(Point(x, y));
                    if(xmin>x) xmin=x;
                    if(xmax<x) xmax=x;
                }
            }
        }
    }


    double A=0,//sigma of (xi*xi)
            B=0,//sigma of (xi)
            C=0,//sigma of (xi*yi)
            D=0;//sigma of (yi)
    double k2x,// k of ercheng
            b2x;//b of ercheng
    int m=store.size();
    size_t i;
    for(i=0; i<store.size(); i++){
        double tx=store[i].x;
        double ty=store[i].y;
        A+=tx*tx;
        B+=tx;
        C+=tx*ty;
        D+=ty;
    }
    k2x=(D*B-m*C)/(B*B-m*A);
    b2x=(D-B*k2x)/m;

  //  line( im2x, Point(xmin, k2x*xmin+b2x), Point(xmax, k2x*xmax+b2x), Scalar(255), 3, CV_AA);
   // line( img, Point(xmin, k2x*xmin+b2x), Point(xmax, k2x*xmax+b2x), Scalar(255), 3, CV_AA);
  //  imshow("im2x", im2x);
    imshow("imgg", imgg);
    imshow("img", img);

    Mat im2x=imgg.clone();
    cout << "xmin=" << xmin << "    xmax=" << xmax << endl;
    line( im2x, Point(xmin, k2x*xmin+b2x), Point(xmax, k2x*xmax+b2x), Scalar(255), 3, CV_AA);
   //line( im2x, Point(0, 0), Point(100, 100), Scalar(0,0,255), 3, CV_AA);
    imshow("im2x", im2x);

    waitKey();

    return 0;
}

  

posted @ 2013-06-01 13:08  ChrisZZ  阅读(2863)  评论(0编辑  收藏  举报