C#使用opencv截取旋轉(zhuǎn)矩形區(qū)域圖像的實(shí)現(xiàn)示例
前言:最近在工程中需要用到截取RotatedRect中的圖形,保存為Mat做后續(xù)處理。發(fā)現(xiàn)opencv文檔中沒(méi)有這個(gè)api,最開(kāi)始想到的方案是將整張圖片進(jìn)行相應(yīng)的旋轉(zhuǎn),然后再?gòu)闹薪厝≌匦危俏覀円@取的是部分區(qū)域,將整張圖片進(jìn)行旋轉(zhuǎn)會(huì)造成很多的資源浪費(fèi)。所以需要自行實(shí)現(xiàn)一個(gè)旋轉(zhuǎn)矩形的方案。
實(shí)現(xiàn)方法
原理是利用向量空間變換,如圖

通過(guò)
Point2f[] points=rotatedRect.Points();
可獲取RotatedRect四個(gè)端點(diǎn)的坐標(biāo)。先選定一個(gè)點(diǎn)為輸出Mat左上端點(diǎn),這里采取的是離原圖左上角最近的端點(diǎn)(即x2+y2最小)。然后再選取相鄰的2個(gè)端點(diǎn),作向量Vx及Vy。這里可以根據(jù)需求來(lái)選定Vx和Vy的方向,比如要求輸出Mat的Width大于Height,那就選定長(zhǎng)的一邊為Vx。這里直接選定左上端點(diǎn)到順時(shí)針?lè)较虻亩它c(diǎn)的向量為Vx,即如圖所示。
在選定好Vx和Vy后,進(jìn)行向量空間的變換。設(shè)在輸出的Mat中任一一個(gè)坐標(biāo)點(diǎn)(i,j),對(duì)應(yīng)在輸出Mat中的任一坐標(biāo)點(diǎn)(x,y)。設(shè)左上端點(diǎn)坐標(biāo)(x0,y0) (即圖中點(diǎn)2),
滿足:
設(shè)目標(biāo)RotatedRect長(zhǎng)寬為width height,有

然后做一個(gè)二重循環(huán),將j從0循環(huán)到height,i從0循環(huán)到width,就可以得到輸出Mat所有像素的信息。
下面為一個(gè)截取BGR類(lèi)型的Mat的RotatedRect的代碼
///<Summary>
///利用向量運(yùn)算截取一個(gè)RotatedRect區(qū)域
///</Summary>
///<param name="img">類(lèi)型為CV_U8C3的Mat</param>
///<param name="rotatedRect">RotatedRect</param>
public static Mat sliceRotetedImg8UC3(Mat img,RotatedRect rotatedRect){
// Rect bounding=rotatedRect.BoundingRect();
Point2f[] points=rotatedRect.Points();
int topLeftIndex=0;
double topLeftR=points[0].X*points[0].X+points[0].Y*points[0].Y;
for(int i=1;i<4;i++){
double r=points[i].X*points[i].X+points[i].Y*points[i].Y;
if(r<topLeftR){
topLeftIndex=i;
topLeftR=r;
}
}
double x1=points[(topLeftIndex+1)%4].X-points[topLeftIndex].X,y1=points[(topLeftIndex+1)%4].Y-points[topLeftIndex].Y;
double x2=points[(topLeftIndex+3)%4].X-points[topLeftIndex].X,y2=points[(topLeftIndex+3)%4].Y-points[topLeftIndex].Y;
double vX1=x1,vY1=y1,vX2=x2,vY2=y2;
int width=(int)Math.Sqrt(vX1*vX1+vY1*vY1),height=(int)Math.Sqrt(vX2*vX2+vY2*vY2);
Mat ret=new Mat(new Size(width,height),MatType.CV_8UC3);
// Console.WriteLine($"width={width},height={height}");
var indexer1=img.GetGenericIndexer<Vec3b>();
var indexer2=ret.GetGenericIndexer<Vec3b>();
for(int j=0;j<ret.Height;j++){
for(int i=0;i<ret.Width;i++){
double kx=(double)i/width,ky=(double)j/height;
int x=(int)(points[topLeftIndex].X+kx*vX1+ky*vX2),y=(int)(points[topLeftIndex].Y+kx*vY1+ky*vY2);
indexer2[j,i]=indexer1[y,x];
}
}
return ret;
}
到此這篇關(guān)于C#使用opencv截取旋轉(zhuǎn)矩形區(qū)域圖像的文章就介紹到這了,更多相關(guān)C# opencv截取內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- c# WinForm制作圖片編輯工具(圖像拖動(dòng)、縮放、旋轉(zhuǎn)、摳圖)
- C# 使用 GDI+ 實(shí)現(xiàn)添加中心旋轉(zhuǎn)(任意角度)的文字
- 利用C#代碼實(shí)現(xiàn)圖片旋轉(zhuǎn)360度
- C#實(shí)現(xiàn)計(jì)算一個(gè)點(diǎn)圍繞另一個(gè)點(diǎn)旋轉(zhuǎn)指定弧度后坐標(biāo)值的方法
- C#中圖片旋轉(zhuǎn)和翻轉(zhuǎn)(RotateFlipType)用法分析
- C#控制圖像旋轉(zhuǎn)和翻轉(zhuǎn)的方法
- C# VTK 移動(dòng)旋轉(zhuǎn)交互功能實(shí)現(xiàn)
相關(guān)文章
基于C#編寫(xiě)一個(gè)遠(yuǎn)程桌面應(yīng)用
封閉環(huán)境無(wú)法拷貝外來(lái)的遠(yuǎn)程桌面軟件,所以這篇文章小編就來(lái)帶大家用C#編寫(xiě)一個(gè)簡(jiǎn)單的遠(yuǎn)程桌面應(yīng)用,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-10-10
WPF+SkiaSharp實(shí)現(xiàn)自繪彈幕效果
這篇文章主要為大家詳細(xì)介紹了如何利用WPF和SkiaSharp實(shí)現(xiàn)自制彈幕效果,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)或工作有一定幫助,感興趣的小伙伴可以了解一下2022-09-09
C#修改IIS站點(diǎn)framework版本號(hào)的方法
這篇文章主要介紹了C#修改IIS站點(diǎn)framework版本號(hào)的方法,涉及C#調(diào)用使用ASP.NET IIS注冊(cè)工具Aspnet_regiis.exe進(jìn)行IIS站點(diǎn)framework版本號(hào)修改的方法,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10
C#使用正則表達(dá)式隱藏手機(jī)號(hào)中間四位為*
這篇文章主要介紹了C#使用正則表達(dá)式隱藏手機(jī)號(hào)中間四位為*的相關(guān)資料,需要的朋友可以參考下2017-06-06
C# List<T> Contains<T>()的用法小結(jié)
本篇文章主要是對(duì)C#中List<T> Contains<T>()的用法進(jìn)行了總結(jié)介紹,需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助2014-01-01

