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

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

wptr33 2025-08-02 22:18 30 浏览

需求

使用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),除非设置的分辨率正好是摄像头本身支持。

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

解决方法:

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

相关推荐

oracle数据导入导出_oracle数据导入导出工具

关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...

继续学习Python中的while true/break语句

上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个...

python continue和break的区别_python中break语句和continue语句的区别

python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...

简单学Python——关键字6——break和continue

Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...

2-1,0基础学Python之 break退出循环、 continue继续循环 多重循

用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...

Python 中 break 和 continue 傻傻分不清

大家好啊,我是大田。...

python中的流程控制语句:continue、break 和 return使用方法

Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...

L017:continue和break - 教程文案

continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...

作为前端开发者,你都经历过怎样的面试?

已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...

面试被问 const 是否不可变?这样回答才显功底

作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...

2023金九银十必看前端面试题!2w字精品!

导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。...

前端面试总结_前端面试题整理

记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...

由浅入深,66条JavaScript面试知识点(七)

作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录...

2024前端面试真题之—VUE篇_前端面试题vue2020及答案

添加图片注释,不超过140字(可选)...

今年最常见的前端面试题,你会做几道?

在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...