




版權(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ì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 紡織品測試方法的試題及答案
- 客戶合同協(xié)議書
- 林地合同協(xié)議書
- 計(jì)生合同協(xié)議書
- 鉗工高級(jí)面試題及答案
- 餐廳兩人合同協(xié)議書
- 協(xié)議書合同規(guī)范
- 歷年中考英語2014甘肅蘭州英語試卷+答案+解析
- 轉(zhuǎn)讓電費(fèi)合同協(xié)議書
- 合同協(xié)議書教學(xué)
- NB/T 11643-2024煤炭快速定量裝車系統(tǒng)通用技術(shù)標(biāo)準(zhǔn)
- 2025年電子信息工程專業(yè)考試卷及答案
- 網(wǎng)絡(luò)輿情的實(shí)時(shí)監(jiān)測與分析-全面剖析
- 北京2025年中國環(huán)境監(jiān)測總站招聘(第二批)筆試歷年參考題庫附帶答案詳解
- 定額〔2025〕1號(hào)文-關(guān)于發(fā)布2018版電力建設(shè)工程概預(yù)算定額2024年度價(jià)格水平調(diào)整的通知
- 2021年妊娠期血壓管理中國專家共識(shí)
- 一種基于STM32的智能門鎖系統(tǒng)的設(shè)計(jì)-畢業(yè)論文
- 上海建設(shè)工程通用硅酸鹽水泥質(zhì)量檢驗(yàn)報(bào)告 - 上海水泥行業(yè)協(xié)會(huì)
- 餐飲服務(wù)六大技能培訓(xùn).ppt
- 基于單片機(jī)的電話撥號(hào)系統(tǒng)的設(shè)計(jì)畢業(yè)論文
- 交通工程設(shè)施設(shè)計(jì)設(shè)計(jì)報(bào)告
評(píng)論
0/150
提交評(píng)論