У меня есть изображение, преобразованное в CvMat
Matrix say CVMat source
. Как только я получу интересующую область из source
, я хочу, чтобы остальная часть алгоритма была применена только к интересующей области. Для этого я думаю, что мне придется как-то обрезать матрицу source
, которую я не могу сделать. Есть ли способ или функция, которые могут обрезать матрицу CvMat
и возвращать другую обрезанную матрицу CvMat
? спасибо.
Как обрезать CvMat в OpenCV?
Ответ 1
OpenCV обладает областью интересов, которую вы можете найти полезной. Если вы используете cv::Mat
, вы можете использовать что-то вроде следующего.
// You mention that you start with a CVMat* imagesource
CVMat * imagesource;
// Transform it into the C++ cv::Mat format
cv::Mat image(imagesource);
// Setup a rectangle to define your region of interest
cv::Rect myROI(10, 10, 100, 100);
// Crop the full image to that image contained by the rectangle myROI
// Note that this doesn't copy the data
cv::Mat croppedImage = image(myROI);
Ответ 2
Я знаю, что этот вопрос уже решен.. но есть очень простой способ обрезать. вы можете просто сделать это в одной строке -
Mat cropedImage = fullImage(Rect(X,Y,Width,Height));
Ответ 3
Чтобы получить лучшие результаты и устойчивость к различным типам матриц, вы можете сделать это в дополнение к первому ответу, который копирует данные:
cv::Mat source = getYourSource();
// Setup a rectangle to define your region of interest
cv::Rect myROI(10, 10, 100, 100);
// Crop the full image to that image contained by the rectangle myROI
// Note that this doesn't copy the data
cv::Mat croppedRef(source, myROI);
cv::Mat cropped;
// Copy the data into new matrix
croppedRef.copyTo(cropped);
Ответ 4
Чтобы создать копию нужного урожая, мы можем сделать следующее:
// Read img
cv::Mat img = cv::imread("imgFileName");
cv::Mat croppedImg;
// This line picks out the rectangle from the image
// and copies to a new Mat
img(cv::Rect(xMin,yMin,xMax-xMin,yMax-yMin)).copyTo(croppedImg);
// Display diff
cv::imshow( "Original Image", img );
cv::imshow( "Cropped Image", croppedImg);
cv::waitKey();
Ответ 5
Я понимаю, что на этот вопрос был дан ответ, но, возможно, это может быть полезно кому-то...
Если вы хотите скопировать данные в отдельный объект cv:: Mat, вы можете использовать функцию, аналогичную этой:
void ExtractROI(Mat& inImage, Mat& outImage, Rect roi){
/* Create the image */
outImage = Mat(roi.height, roi.width, inImage.type(), Scalar(0));
/* Populate the image */
for (int i = roi.y; i < (roi.y+roi.height); i++){
uchar* inP = inImage.ptr<uchar>(i);
uchar* outP = outImage.ptr<uchar>(i-roi.y);
for (int j = roi.x; j < (roi.x+roi.width); j++){
outP[j-roi.x] = inP[j];
}
}
}
Важно отметить, что это будет работать только на одноканальных изображениях.
Ответ 6
Вы можете легко обрезать Mat, используя opencv funtions.
setMouseCallback("Original",mouse_call);
mouse_call
приведен ниже:
void mouse_call(int event,int x,int y,int,void*)
{
if(event==EVENT_LBUTTONDOWN)
{
leftDown=true;
cor1.x=x;
cor1.y=y;
cout <<"Corner 1: "<<cor1<<endl;
}
if(event==EVENT_LBUTTONUP)
{
if(abs(x-cor1.x)>20&&abs(y-cor1.y)>20) //checking whether the region is too small
{
leftup=true;
cor2.x=x;
cor2.y=y;
cout<<"Corner 2: "<<cor2<<endl;
}
else
{
cout<<"Select a region more than 20 pixels"<<endl;
}
}
if(leftDown==true&&leftup==false) //when the left button is down
{
Point pt;
pt.x=x;
pt.y=y;
Mat temp_img=img.clone();
rectangle(temp_img,cor1,pt,Scalar(0,0,255)); //drawing a rectangle continuously
imshow("Original",temp_img);
}
if(leftDown==true&&leftup==true) //when the selection is done
{
box.width=abs(cor1.x-cor2.x);
box.height=abs(cor1.y-cor2.y);
box.x=min(cor1.x,cor2.x);
box.y=min(cor1.y,cor2.y);
Mat crop(img,box); //Selecting a ROI(region of interest) from the original pic
namedWindow("Cropped Image");
imshow("Cropped Image",crop); //showing the cropped image
leftDown=false;
leftup=false;
}
}
Подробнее см. ссылку Обрезка изображения с помощью мыши