百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

使用OpenCV库操作摄像头拍照、调节参数和视频录制

zhezhongyun 2025-04-02 00:26 9 浏览

需求

使用OpenCV做功能,播放摄像头(usb和网络),对摄像头设备进行参数调整(亮度、对比度、饱和度、色调、增益、曝光度)调节,拍照和录像。

原理

使用OpenCV打开摄像头(可打开USB和网路哦摄像头),渲染图像显示,可使用OpenCV属性调整摄像头的各项参数,使用拍照可以将当前图片拍照,使用录像可以从当前时间点开始录像直至停止录像

注意

目前测试,即使PC上有编码器,但是OpenCV存储mat为对应的录像视频文件失败,出现:

录制完视频大小为200多B(基本为0),mp4格式时(查看入坑一)

录制完视频大小为6KB,avi格式时

录制avi传入图像mat,源码内部出现错误宕机

运行效果:

核心代码

打开摄像头代码

bool OpenCVManager::startCapture(int usb, int width, int height)
{
 if(!_pVideoCapture->open(usb))
 {
 qDebug() << __FILE__ << __LINE__ << "Failed to start capture :" << usb return false _width='width;' _height='height;' _pvideocapture->set(CV_CAP_PROP_FRAME_WIDTH, _width);
 _pVideoCapture->set(CV_CAP_PROP_FRAME_HEIGHT, _height);
 _width = _pVideoCapture->get(CV_CAP_PROP_FRAME_WIDTH);
 _height = _pVideoCapture->get(CV_CAP_PROP_FRAME_HEIGHT);
 _pVideoCapture->set(CV_CAP_PROP_FPS, 25);
 _brightness = _pVideoCapture->get(cv::CAP_PROP_BRIGHTNESS);
 _contrast = _pVideoCapture->get(cv::CAP_PROP_CONTRAST);
 _saturation = _pVideoCapture->get(cv::CAP_PROP_SATURATION);
 _hue = _pVideoCapture->get(cv::CAP_PROP_HUE);
 _gain = _pVideoCapture->get(cv::CAP_PROP_GAIN);
 _exposure = _pVideoCapture->get(cv::CAP_PROP_EXPOSURE);
 QTimer::singleShot(0, this, SLOT(slot_captrueFrame()));
 return true;
}

调整属性代码

bool OpenCVManager::getShowProperty() const
{
 return _showProperty;
}

void OpenCVManager::setShowProperty(bool value)
{
 _showProperty = value;
}

double OpenCVManager::getBrightness()
{
 return _brightness;
}

void OpenCVManager::setBrightness(double value)
{
 _brightness = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_BRIGHTNESS, _brightness);
 }
}

double OpenCVManager::getContrast() const
{
 return _contrast;
}

void OpenCVManager::setContrast(double value)
{
 _contrast = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_CONTRAST, _contrast);
 }
}

double OpenCVManager::getSaturation() const
{
 return _saturation;
}

void OpenCVManager::setSaturation(double value)
{
 _saturation = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_SATURATION, _saturation);
 }
}

double OpenCVManager::getHue() const
{
 return _hue;
}

void OpenCVManager::setHue(double value)
{
 _hue = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_HUE, _hue);
 }
}

double OpenCVManager::getGain() const
{
 return _gain;
}

void OpenCVManager::setGain(double value)
{
 _gain = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_GAIN, _gain);
 }
}

double OpenCVManager::getExposure() const
{
 return _exposure;
}

void OpenCVManager::setExposure(double value)
{
 _exposure = value;
 if(_pVideoCapture)
 {
 _pVideoCapture->set(cv::CAP_PROP_EXPOSURE, _exposure);
 }
}

拍照代码

void PhotoAndRecordWidget::on_pushButton_photo_clicked()
{
 QString timeStr = QDateTime::currentDateTime().toString("yyyy-MM-hh hh_mm_ss");
 QString dirName = "photos";
 if(!QFile::exists(dirName))
 {
 QDir dir;
 if(!dir.mkdir(dirName))
 {
 ui->label_state->setText("创建文件夹 " + dirName + "失败!!!");
 }
 }
 QString filePath = QString("%1/%2.png").arg(dirName).arg(timeStr);
 if(_image.save(filePath))
 {
 ui->label_state->setText("保存照片至: " + filePath);
 }else{
 ui->label_state->setText("保存照片失败!!!");
 }
}

录像代码

开启录像

void OpenCVManager::startRecord(QString pathFile)
{
 // 多线程处理
 QMetaObject::invokeMethod(this, "slot_startRecord",
 Qt::DirectConnection, Q_ARG(QString, pathFile));
}
void OpenCVManager::slot_startRecord(QString filePath)
{
 if(_pVideoWrite)
 {
 qDebug() << __FILE__ << __LINE__ << "It's recording!!!";
 return;
 }
 _pVideoWrite = new cv::VideoWriter;
 QString ext = filePath.mid(filePath.lastIndexOf(".") + 1);
 int cvFourcc = 0;
 if(ext == "mpg")
 {
 cvFourcc = CV_FOURCC('D','I','V','X');
 qDebug() << __FILE__ << __LINE__<< ext << "DIVX" << cvFourcc;
 }else if(ext == "avi")
 {
 cvFourcc = CV_FOURCC('M','J','P','G');
 qDebug() << __FILE__ << __LINE__<< ext << "MJPG" << cvFourcc;
 }else if(ext == "mp4")
 {
 // mp4目前录制不成功(可以生成文件,但是打开失败)
// cvFourcc = CV_FOURCC('M','P','4','2');
 cvFourcc = CV_FOURCC('M','J','P','G');
// cvFourcc = CV_FOURCC('I', 'Y', 'U', 'V');
 qDebug() << __FILE__ << __LINE__<< ext << "MP42" << cvfourcc _pvideowrite->open(filePath.toStdString(), cvFourcc, 25, cv::Size(_width, _height));

 _recordFilePath = filePath;
 _recording = true;
}

录像过程

void OpenCVManager::slot_captrueFrame()
{
 if(!_running)
 {
 return;
 }
 if(_pVideoCapture->isOpened())
 {
 cv::Mat mat;
 *_pVideoCapture >> mat;
 if(_showProperty)
 {
 cv::putText(mat, QString("brightness: %1").arg(_brightness).toStdString(),
 cvPoint(0, 30), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString(" contrast: %1").arg(_contrast ).toStdString(),
 cvPoint(0, 60), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString("saturation: %1").arg(_saturation).toStdString(),
 cvPoint(0, 90), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString(" hue: %1").arg(_hue ).toStdString(),
 cvPoint(0, 120), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString(" gain: %1").arg(_gain ).toStdString(),
 cvPoint(0, 150), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString(" exposure: %1").arg(_exposure ).toStdString(),
 cvPoint(0, 180), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 cv::putText(mat, QString("press ESC out").toStdString(),
 cvPoint(0, 210), cv::FONT_HERSHEY_COMPLEX, 1.0, cv::Scalar(255));
 }
 if(_recording)
 {
 _pVideoWrite->write(mat);
 }
 emit signal_captureOneFrame(mat);
 QTimer::singleShot(10, this, SLOT(slot_captrueFrame()));
 }
}

关闭录像

void OpenCVManager::stopRecord()
{
 // 多线程处理
 QMetaObject::invokeMethod(this, "slot_stopRecord", Qt::DirectConnection);
}

void OpenCVManager::slot_stopRecord()
{
 if(_pVideoWrite)
 {
 _recording = false;
 _pVideoWrite->release();
 delete _pVideoWrite;
 _pVideoWrite = 0;
 }
}

入坑记录

入坑一:录制视频保存为空

解决方法:

编解码器得问题,cv::VideoWrite查阅相关资料发现其只支持固定的几个格式,其中就包括avi。

入坑二:录制视频奔溃

原因:

因为初始设置摄像头的宽高(400 x 400),根据测试推断摄像头会默认给最接近初始化设置的分辨率,但是却不是直接是设置的(400 x 400)而是返回了最接近的分辨率(320 x 240),除非设置的分辨率正好是摄像头本身支持。

所以设置分辨率是需要摄像头硬件支持。

解决方法:

进一步验证同时解决该问题

相关推荐

字体缩放(方式一)(字体缩放150%怎么做)

通过元素宽度和字数计算得到缩放简单实现如下:/***字体最大为视觉要求大小(maxFontSize);超出缩小字体显示,最小为minFontSize;最小字体时超出部分使用圆点(...);*p...

网页世界隐藏的神秘代码语言,竟能这样改变布局

CSS基础:选择器与属性CSS(CascadingStyleSheets)是用于控制网页外观的一门样式表语言。它通过定义HTML元素的显示方式来增强网页的表现力。CSS的选择器允许开发者精确地定位...

CSS属性值计算过程详解(css属性用来定义元素计算)

在CSS中,即使某些属性没有显式声明,浏览器也会通过**属性值计算过程**为每个元素的所有属性赋予最终值。这一过程分为四个关键步骤,以下将逐一解析。1.确定声明值浏览器首先检查所有**直接应用**到...

软网推荐:找回调整Windows 10字号功能

之前的系统,从WindowsXP到早期版本的Windows10,均有字体大小调整功能,但从创意者版Windows10以来,取消了之前的设置选项,取而代之的是自定义缩放比例设置。使用这个功能调整过...

Excel中如何设置文本框属性,实例代码讲解

Excel不仅可以对数据进行处理,而且也可以图形化数据,直观显示数据表达的内容。本节介绍一个很重要的对象,Characters,字符对象,使用Characters对象可修改包含在全文本字符串中的任...

CSS 字体样式(css中字体)

本节我们来讲字体样式,之前我们学习HTML的时候学过一些用于字体加粗、倾斜的标签,但是使用标签来实现的效果肯定没有我们通过CSS中的样式来的方便。接下来我们会给大家介绍下面这几个属性的使用:通...

PC网站建设必备代码知识:HTML基础与应用技巧

在PC网站建设的相关课程里,代码扮演着至关重要的角色。只有熟练运用正确的代码,我们才能打造出功能完善、用户体验出色的PC网站。接下来,我会详细讲解在PC网站建设环节中必须了解的代码知识。HTML基础代...

让你大跌眼镜的疯狂 HTML 和 CSS 技巧

今天,分享一个让你大开眼界的技巧。通过使用这个技巧,你可以将整个网页变成一个CSS编辑器。没错,你从未见过这种方法。当我第一次尝试时,我完全被震惊到了。现在,让我们开始吧!步骤1首先,创建一个基础的...

jQuery EasyUI使用教程:创建一个链接按钮

jQueryEasyUI最新版下载>本教程主要为大家展示如何使用jQueryEasyUI创建一个链接按钮。通常情况下,使用“button/”元素来创建一个按钮;使用“a/”元素来创建链接按钮...

React 19 有哪些新特性?(react100)

如果你对React18还不熟悉,欢迎阅读之前的文章《React18全览[1]》最近React发布了V19RC版本,按照惯例,我们对React19的新特性进行一次深度的体验学习...

Java注解探秘:为什么@PostConstruct能解决你的初始化难题?

你是否在Spring项目中遇到过这样的困扰:明明依赖注入已经完成,但某些配置就是无法正常加载?手动调用初始化方法又容易引发空指针异常?这就是@PostConstruct注解大显身手的时候了!@Post...

AI驱动的表单自动填写(ai置入表格)

我们都同意,填写表格是一项枯燥且耗时的任务。如果我们可以创建一个可以为我们填写表格的AI助手,让我们将时间投入到更有建设性的任务中,那会怎样?AI助手将能够通过调用以表单字段为参数的函数来填写表...

从零到一:小程序设计新手如何快速上手?

开发环境搭建对于小程序设计新手而言,搭建合适的开发环境是首要任务。以小程序为例,其官方提供了功能强大的开发工具——开发者工具。首先,新手需前往官方开发者平台,在页面中找到“工具下载”板块,根据...

JavaSwingGUI从小白到大神-6(续)(java从小白到大牛怎么样)

接上一篇《JavaSwingGUI从小白到大神-6》,因本篇文章3万多字,头条一篇发不完,只能分开发。同事查询面板:CompanyFind.javapublicclassCompanyFind{...

C# winform界面假死(c#程序假死)

针对C#WinForm界面假死问题,以下是分步解决方案:1.使用异步编程(async/await)将耗时操作移至后台线程,保持UI线程响应。步骤:将事件处理函数标记为async。使用Task....