C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件_第1頁
C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件_第2頁
C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件_第3頁
C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件_第4頁
C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件_第5頁
已閱讀5頁,還剩4頁未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡介

第C++OpenCV實(shí)戰(zhàn)之文檔照片轉(zhuǎn)換成掃描文件MatresultImg;//保存處理后圖像

resize(image,resultImg,Size(processWith,peocessHeight));

returnresultImg;

這里再定義一個(gè)顯示圖像的方法:

//顯示圖片

voidvisualize(StringwinName,Matimage){

namedWindow(winName,WINDOW_NORMAL);

imshow(winName,image);

waitKey(0);

2、創(chuàng)建直線類并計(jì)算兩條直線的交點(diǎn)

先定義一個(gè)直線類,包含兩個(gè)端點(diǎn);

假設(shè)已知兩條直線上的兩點(diǎn),怎么求得交點(diǎn)呢?

可以參考這個(gè)網(wǎng)址中的數(shù)學(xué)公式:/wiki/Line%E2%80%93line_intersection

//返回兩條直線的交點(diǎn)

Point2flinesIntersect(Linelin1,Linelin2){

//這里直接根據(jù)網(wǎng)上的數(shù)學(xué)公式求得

intx1=lin1.p1.x,y1=lin1.p1.y;

intx2=lin1.p2.x,y2=lin1.p2.y;

intx3=lin2.p1.x,y3=lin2.p1.y;

intx4=lin2.p2.x,y4=lin2.p2.y;

floatD=(x1-x2)*(y3-y4)-(y1-y2)*(x3-x4);

if(fabs(D)1e-6){

returnPoint2f(

((x1*y2-y1*x2)*(x3-x4)-(x1-x2)*(x3*y4-y3*x4))/D,

((x1*y2-y1*x2)*(y3-y4)-(y1-y2)*(x3*y4-y3*x4))/D);

returnPoint2f(-1,-1);

3、圖像邊緣檢測Canny

Matcanny,gray;

cvtColor(image,gray,COLOR_BGR2GRAY);

doublethreshold_low=threshold(gray,canny,0,255,THRESH_BINARY|cv::THRESH_OTSU);

Canny(gray,canny,threshold_low,threshold_low*2);

returncanny;

注意:進(jìn)行Canny邊緣檢測的效果和圖像的大小有關(guān),可以適當(dāng)對(duì)大圖進(jìn)行縮放;

4、通過霍夫變換進(jìn)行直線檢測

檢測到的直線分為兩類,一類是水平線,一類是豎直線;

為了得到外邊緣框的直線,可以先根據(jù)直線的外接矩形長寬比分為水平和豎直線,再根據(jù)中點(diǎn)的位置,找到邊緣直線;

如上圖所示,如果xy,則將該直線分為水平線,如果yx,則將該直線劃分為水平線;

隨后再根據(jù)中心點(diǎn)的坐標(biāo)大小確定邊緣線;

//進(jìn)行霍夫直線檢測,保存所有檢測到的直線,并且確保直線大于4條

vectorVec4ilines;

//這里的參數(shù)需要根據(jù)圖像大小等因素進(jìn)行微調(diào)

HoughLinesP(canny,lines,1,CV_PI/180,80,height/5,200);

if(lines.size()4){

cout"Findonly"lines.size()"lines,returndirectly"endl;

//將直線分為水平線和垂直線

vectorLinehorizontals,verticals;

Mattmp=image.clone();

for(autov:lines){

doublew=fabs(v[0]-v[2]),h=fabs(v[1]-v[3]);

Linetmp_line(Point(v[0],v[1]),Point(v[2],v[3]));

if(wh)horizontals.push_back(tmp_line);

elseverticals.push_back(tmp_line);

//下面兩行代碼是實(shí)現(xiàn)繪制直線

//line(tmp,Point(v.val[0],v.val[1]),Point(v.val[2],v.val[3]),Scalar(255,0,0),8);

//visualize("houghtest",tmp);

//確保水平線和垂直線至少有兩條

if(horizontals.size()2||verticals.size()2){

cout"Notenoughedgelines..."endl;

//將水平和垂直線按中心點(diǎn)位置進(jìn)行排序,這里的兩個(gè)排序函數(shù)需要自己實(shí)現(xiàn)

sort(horizontals.begin(),horizontals.end(),cmpHeight);

sort(verticals.begin(),verticals.end(),cmpWidth);

//繪制出找到的直線

line(tmp,horizontals[0].p1,horizontals[0].p2,Scalar(255,0,0),8);

line(tmp,horizontals[horizontals.size()-1].p1,horizontals[horizontals.size()-1].p2,Scalar(255,0,0),8);

line(tmp,verticals[0].p1,verticals[0].p2,Scalar(255,0,0),8);

line(tmp,verticals[verticals.size()-1].p1,verticals[verticals.size()-1].p2,Scalar(255,0,0),8);

visualize("houghtest",tmp);

排序函數(shù)的實(shí)現(xiàn):

//對(duì)水平線進(jìn)行排序

boolcmpHeight(constLinel1,constLinel2){

returnl1.center.yl2.center.y;

//對(duì)垂直線進(jìn)行排序

boolcmpWidth(constLinel1,constLinel2){

returnl1.center.xl2.center.x;

5、求單應(yīng)性矩陣

現(xiàn)在已知圖像上目標(biāo)的四個(gè)點(diǎn)坐標(biāo),通過點(diǎn)對(duì)關(guān)系,求得二者之間的單應(yīng)性變換矩陣;

intdst_width=1080,dst_height=1920;

vectorPoint2fdst_pts;

dst_pts.push_back(Point(0,0));

dst_pts.push_back(Point(dst_width-1,0));

dst_pts.push_back(Point(0,dst_height-1));

dst_pts.push_back(Point(dst_width-1,dst_height-1));

MatwarpedImg=Mat::zeros(dst_height,dst_width,CV_8UC3);

Mathomo=getPerspectiveTransform(ori_pts,dst_pts);

warpPerspective(image,warpedImg,homo,warpedImg.size());

visualize("result",warpedImg);

結(jié)果圖:

6、降噪和二值化

降噪采用中值濾波,閾值過濾采用自適應(yīng)的二值化;

voidpostProcess(Matimage){

medianBlur(image,image,7);

cvtColor(image,image,COLOR_BGR2GRAY);

threshold(image,image,0,255,THRESH_BINARY|cv::THRESH_OTSU);

四、方案二:用戶點(diǎn)選目標(biāo)區(qū)域

方案一是完全基于自動(dòng)化的方式,用戶只需要傳入圖像,程序會(huì)自動(dòng)選擇合適的區(qū)域;

優(yōu)點(diǎn)在于其節(jié)省了用戶的人工成本,使得程序更為簡便;

缺點(diǎn)在于算法具有局限性,對(duì)背景及區(qū)域選取有要求,比如不能在區(qū)域外出現(xiàn)干擾物體,也無法滿足用戶的一些特別需求,比如選定大區(qū)域中的小區(qū)域;

方案二的優(yōu)勢在于其強(qiáng)大的靈活性,用戶可以根據(jù)自己的需求選擇相應(yīng)的區(qū)域,程序?qū)?duì)所選區(qū)域進(jìn)行轉(zhuǎn)換;

1、命令行解析

加入命令行參數(shù)的功能,用戶可以從命令行傳入?yún)?shù);

intmain(intargc,char**argv)

constStringkeys=

"{helphusage||printthismessage}"

"{path|D:/project/OpenCV/card.jpg|pathtofile}"

//采用opencv命令行解析的方式

CommandLineParsermyParser(argc,argv,keys);

Stringpath=myParser.getString("path");

coutpathendl;

2、鼠標(biāo)事件

主要實(shí)現(xiàn)用戶點(diǎn)擊鼠標(biāo)時(shí)的一些交互功能:

//這幾個(gè)參數(shù)為默認(rèn)參數(shù)

voidonMouse(intevent,intx,inty,intflags,void*userdata){

//當(dāng)點(diǎn)數(shù)為四個(gè)時(shí),判斷當(dāng)前用戶鼠標(biāo)選取的拖動(dòng)點(diǎn)是哪一個(gè)

if(srcPts.size()==4){

for(inti=0;ii++){

Point2fv=srcPts[i];

if((event==EVENT_LBUTTONDOWN)(abs(v.x-x)20)(abs(v.y-y)20)){

isDragging=true;

drag_index=i;

//用戶點(diǎn)擊鼠標(biāo)左鍵時(shí),加入點(diǎn)

elseif(event==EVENT_LBUTTONDOWN){

srcPts.push_back(Point2f(x,y));

//取消拖動(dòng)

if(event==EVENT_LBUTTONUP){

isDragging=false;

//如果鼠標(biāo)移動(dòng)并且一直按著,就改變?cè)瓉淼狞c(diǎn)

if((event==EVENT_MOUSEMOVE)isDragging){

srcPts[drag_index].x=x;

srcPts[drag_index].y=y;

3、主函數(shù)實(shí)現(xiàn)

定義了鼠標(biāo)函數(shù)之后,需要將其中的操作在窗口進(jìn)行展示:

namedWindow(winNameOri,WINDOW_NORMAL);

namedWindow(winNameRes,WINDOW_NORMAL);

setMouseCallback(winNameOri,onMouse,nullptr);

booldone=false;

while(!done){

if(srcPts.size()4){

img=oriImg.clone();

for(inti=0;isrcPts.size();i++){

circle(img,srcPts[i],10,Scalar(255,255,0),5);

putText(img,labels[i].c_str(),srcPts[i],FONT_HERSHEY_SIMPLEX,1,Scalar(255,255,255),2);

imshow(winNameOri,img);

if(srcPts.size()==4){

img=oriImg.clone();

for(inti=0;ii++){

circle(img,srcPts[i],10,Scalar(255,255,0),5);

line(img,srcPts[i],srcPts[(i+1)%4],Scalar(0,255,0),5);

putText(img,labels[i].c_str(),srcPts[i],FONT_HERSHEY_SIMPLEX,1,Scalar(255,255,255),2);

imshow(winNameOri,img);

4、結(jié)果展示

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

評(píng)論

0/150

提交評(píng)論