opencv模拟景深效果

简介

  在htc的相机应用中发现了一个叫景深的拍照场景,然后练习着用opencv来模拟实现。


hcl景深场景效果

                              


本例实现效果

                                     


实现过程

  要实现本例,需要用到两方面知识:1、图像的高斯模糊。 2.在图像中去出感兴趣的圆形区域。


高斯模糊

  用的是opencv函数:cvSmooth。该函数参数具体使用,看opencv官网的api介绍吧:http://docs.opencv.org/modules/refman.html
#include <opencv2/core/core.hpp>                                                                                                   
#include <opencv2/highgui/highgui.hpp>
#include <math.h>
#include <string.h>
#include <opencv/cv.h>
#include <stdio.h>
 
using namespace cv;
 
int width, height;
char pic_name[20];
Mat mat1;
 
void mySmooth(Mat mat1, Mat mat2, int flag, int width, int height){
    IplImage pI1 = mat1;    
    IplImage pI2 = mat2;
 
    cvSmooth(&pI1, &pI2, flag, width, height);
}
 
int main(int argc,char *argv[]){
    memcpy(pic_name,argv[1], sizeof(argv[1]));
    mat1 = imread(pic_name, 1);
 
    imshow("1", mat1);
    mySmooth(mat1, mat1,CV_GAUSSIAN, 23, 23);                                                                                      
    imshow("2", mat1);
    cv::waitKey(0);
    return 0;
}
  显示效果如下:
                                


圆形兴趣区域

  在opencv的ROI中,最常使用的在图片中截取出感兴趣区域方式如下:
cv::Mat imageROI;
imageROI = img(cv::Rect(40,40,40,40));
  该代码表示,在图像img的坐标(40,40)位置,取出一个width:40,height:40的矩形复制到imageROI中。
但是如果我们想复制取出的区域为圆形的话,就不能用这个办法。需要使用如下方式:
/**************取出圆形感兴趣区域**********/
    mat1 = imread(pic_name, 1);                                                                                                    
    src = mat1;
    res = cvCreateImage(cvGetSize(&src), 8, 3);
    roi = cvCreateImage(cvGetSize(&src), 8, 1);
 
    cvZero(roi);
    cvZero(res);
    cvCircle(roi,cvPoint(50, 50), 30,CV_RGB(255, 255, 255),-1, 8, 0);
    cvAnd(&src, &src, res, roi);    
    /******************************************/
  首先新建两个和原图像src相同大小的Mat:res、roi,然后将res和roi内容都清空,接着在roi中坐标(50,50)为圆形,30为半径画一个圆。
接着调用cvAnd函数中,将roi作为掩码输入,就在res中复制得到了src对应在roi画圆区域的图像内容。
  具体的效果演示如下:
                                       


本例代码

  本例使用的代码主要就是如上知识,另外还有鼠标拖动来控制图像景深圆圈的大小和开始位置。
具体代码如下:
#include <opencv2/core/core.hpp>                                                                                                   
#include <opencv2/highgui/highgui.hpp>
#include <math.h>
#include <string.h>
#include <opencv/cv.h>
#include <stdio.h>
 
using namespace cv;
 
int width, height;
int pic_info[3];
char pic_name[20];
Mat mat1;
IplImage src, *res, *roi;
 
void mySmooth(Mat mat1, Mat mat2, int flag, int width, int height){
	IplImage pI1 = mat1;	
	IplImage pI2 = mat2;
 
	cvSmooth(&pI1, &pI2, flag, width, height);
}
 
void mypic_merge(IplImage* src, IplImage* res, IplImage* dst){
	CvScalar s;
	int height = src->height;
	int width = src->width;
	int i, j;
 
	for(i=0;i<height;i++){
		for(j=0;j<width;j++){
			s = cvGet2D(res, i, j);
			if((s.val[0] == 0) && (s.val[1] == 0) && (s.val[2] == 0)){
				s = cvGet2D(src, i, j);	
			}
			cvSet2D(dst,i,j,s);
		}
	}
}
 
void on_mouse( int event, int x, int y, int flags, void* ustc)  
{
 
	if(event == CV_EVENT_LBUTTONDOWN){
		pic_info[0] = x;   /*width1*/
		pic_info[1] = y;   /*height1*/
		pic_info[2] = 0;   /*width2*/
		mat1=cv::imread(pic_name,1);  
		cv::imshow("1",mat1);
	}
	if(flags == CV_EVENT_FLAG_LBUTTON){
		x = abs(x - pic_info[0]);
		y = abs(y - pic_info[1]);
		pic_info[2] = (int)sqrt((x*x + y*y));
		mat1=cv::imread(pic_name,1);
		src = mat1;
 
		res = cvCreateImage(cvGetSize(&src), 8, 3);
		roi = cvCreateImage(cvGetSize(&src), 8, 1);
 
/**************取出圆形感兴趣区域**********/
		cvZero(roi);
		cvZero(res);
		cvCircle(roi,cvPoint(pic_info[0], pic_info[1]), pic_info[2],CV_RGB(255, 255, 255),-1, 8, 0);
		cvAnd(&src, &src, res, roi);    
/******************************************/
		mySmooth(mat1, mat1,CV_GAUSSIAN, 23, 23);
		mypic_merge(&src, res, res);
 
		cvNamedWindow("2", 1);
		cvShowImage("2", res);
	}
	printf("circle:%d\n", pic_info[2]);
} 
 
int main(int argc,char *argv[]){
	memcpy(pic_name,argv[1], sizeof(argv[1]));
	mat1 = imread(pic_name, 1);
	width = mat1.rows;
	height = mat1.cols;
 
	imshow("1", mat1);
	cvSetMouseCallback("1", on_mouse, NULL);
	cv::waitKey(0);
	return 0;
}
  如上,就能实现在图像窗口"1"中,通过鼠标点击和拖拽,控制景深圆圈变化。
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: OpenCVSharp是一个在C#中使用OpenCV库的封装。而景深融合是一种利用图像处理技术把多个图像的焦点位置进行融合,从而得到具有较大景深范围的图像。 在使用OpenCVSharp进行景深融合时,一般的流程如下: 1. 首先,读取多张具有不同焦点位置的图像。可以利用OpenCVSharp提供的函数来实现图像读取操作。 2. 使用OpenCVSharp中的函数,比如Cv2.GaussianBlur()或Cv2.Laplacian()来对每张图像进行图像增强处理,以提取出图像中的高频信息。 3. 对每张图像进行焦点检测,可以使用OpenCVSharp中的函数,比如Cv2.Canny()或Cv2.Sobel()来检测图像中的边缘信息。 4. 根据焦点检测的结果,计算每张图像的焦点位置。 5. 根据焦点位置,进行图像融合操作。可以使用OpenCVSharp中的函数,比如Cv2.AddWeighted()或Cv2.Blend()来实现图像融合。 6. 最后,将融合后的图像保存或显示出来。 需要注意的是,景深融合是一个复杂且需要一些图像处理知识的任务。在实际应用中,还需要根据具体的需求和场景进行参数调整和算法优化。另外,OpenCVSharp作为OpenCV库的C#封装,其提供了很多常用的图像处理函数和工具,使得在C#中进行图像处理变得方便快捷。但使用OpenCVSharp进行景深融合前,也需要对OpenCVSharp的函数和用法有一定的了解。 ### 回答2: OpenCvSharp景深融合是指通过使用OpenCvSharp库中的函数和方法,将多张具有不同焦点距离的图像进行融合,从而得到一张具有较大景深的图像。 在进行景深融合时,首先需要将输入的图像进行预处理,包括去噪、边缘提取等步骤。然后,利用OpenCvSharp中的函数,如findContours,来识别图像中的轮廓并提取出来。 接下来,通过计算每个轮廓的深度值,确定各个轮廓在图像中的位置。然后,根据轮廓的深度值,使用OpenCvSharp库中的函数和算法,如图像融合、加权平均等,将多个图像进行融合。 通过景深融合,我们可以得到一张具有较大景深的图像,即使在不同距离上的物体都能清晰可见。这对于摄影、机器人视觉等领域具有重要的应用价值。 总结起来,OpenCvSharp景深融合是通过使用OpenCvSharp库中的函数和算法,将多个具有不同焦点距离的图像进行融合,从而得到一张具有较大景深的图像。通过对图像进行预处理、轮廓提取和深度计算等步骤,我们可以获得清晰可见的物体图像,为各个领域的应用提供了有力支持。 ### 回答3: OpenCVSharp景深融合是一种图像处理技术,旨在将具有不同焦点距离的多张图像合成成一张具有较大景深范围的图像。 在传统的摄影中,当拍摄远景时,景深较大,前景和背景都能清晰呈现;当拍摄近景时,景深相对较小,只有部分区域能够清晰呈现。而景深融合技术通过将多张不同焦点距离的图像进行融合,可以实现对整个场景的清晰呈现。 具体实现时,首先需要使用不同的焦点距离对同一场景进行多次拍摄,得到一系列具有不同焦点的图像。接下来,利用OpenCVSharp库中的图像处理函数,可以对这些图像进行处理,提取出每张图像中的清晰区域。 然后,通过将多张清晰区域进行融合,可以得到一张包含整个场景清晰部分的图像。这种融合可以通过图像叠加、像素加权等方式实现。 最后,通过景深融合技术,可以得到一张具有较大景深范围的图像,能够清晰展示整个场景,同时保留了前景和背景的细节。 总之,OpenCVSharp景深融合技术是一种通过融合具有不同焦点距离的图像,实现对整个场景的清晰展示的图像处理技术。这种技术在摄影、计算机视觉等领域有广泛应用,可以提升图像的质量和视觉效果

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值