有一个需求,需要在Linux客户机上,实现调用摄像头(webcam)来捕捉图像。经过研究,可以使用V4L(Video for Linux2)来实现,一个在Linux下通用的视频驱动框架,支持绝大多数的摄像头(外置和内置摄像头都支持),目前有很多基于linux的网络摄像头和图像采集卡都在使用此框架程序。
如果需要更高级的应用开发,可以选择使用Libcamera,基于V4L开发,是作为V4L的上层扩展,解决了V4L接口的复杂性问题,大幅降低了开发难度,为开发者提供更高级、更建议的摄像头开发框架。
V4L
定位:内核驱动接口(用户空间通过ioctl调用)。
架构层级:底层接口,可以直接操作硬件。
目标场景:高性能、低延迟的应用场景(如工业摄像头、实时处理)
依赖关系:仅需内核支持(大部分Linux系统都已内置)
用法范例
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
#include <linux/videodev2.h>
#define DEVICE "/dev/video0"
#define WIDTH 640
#define HEIGHT 480
#define FILENAME "capture.jpg"
int main(){
int fd=open(DEVICE, O_RDWR);
if(fd<0){
perror("无法打开设备");
return -1;
}
//查询设备能力
struct v4l2_capability cap;
if (ioctl(fd, VIDIOC_QUERYCAP, &cap) < 0) {
perror("查询设备能力失败");
close(fd);
return -1;
}
//设置视频格式
struct v4l2_format fmt;
memset(&fmt, 0, sizeof(fmt));
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
fmt.fmt.pix.width = WIDTH;
fmt.fmt.pix.height = HEIGHT;
fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_JPEG; // 直接捕获JPEG格式
if (ioctl(fd, VIDIOC_S_FMT, &fmt) < 0) {
perror("设置格式失败");
close(fd);
return -1;
}
//请求缓冲区
struct v4l2_requestbuffers req;
memset(&req, 0, sizeof(req));
req.count = 1;
req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
req.memory = V4L2_MEMORY_MMAP;
if (ioctl(fd, VIDIOC_REQBUFS, &req) < 0) {
perror("请求缓冲区失败");
close(fd);
return -1;
}
//查询缓冲区并映射
struct v4l2_buffer buf;
memset(&buf, 0, sizeof(buf));
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
buf.index = 0;
if (ioctl(fd, VIDIOC_QUERYBUF, &buf) < 0) {
perror("查询缓冲区失败");
close(fd);
return -1;
}
unsigned char* buffer_start = mmap(NULL, buf.length,
PROT_READ | PROT_WRITE,
MAP_SHARED, fd, buf.offset);
if(buffer_start == MAP_FAILED){
perror("内存映射失败");
close(fd);
return -1;
}
//将缓冲区放入队列
if (ioctl(fd, VIDIOC_QBUF, &buf) < 0) {
perror("将缓冲区放入队列失败");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
//开始捕获
enum v4l2_buf_type type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (ioctl(fd, VIDIOC_STREAMON, &type) < 0) {
perror("开始捕获失败");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
//等待帧就绪(这里简化处理,实际应用中可能需要循环等待)
fd_set fds;
FD_ZERO(&fds);
FD_SET(fd, &fds);
struct timeval tv = {0};
tv.tv_sec = 2;
int r = select(fd + 1, &fds, NULL, NULL, &tv);
if (r < 0) {
perror("select出错");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
else if (r == 0) {
fprintf(stderr, "捕获超时\n");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
//从队列中取出缓冲区
if (ioctl(fd, VIDIOC_DQBUF, &buf) < 0) {
perror("从队列中取出缓冲区失败");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
//保存图像到文件
FILE* file = fopen(FILENAME, "wb");
if (!file) {
perror("无法创建文件");
munmap(buffer_start, buf.length);
close(fd);
return -1;
}
fwrite(buffer_start, buf.bytesused, 1, file);
fclose(file);
//停止捕获并释放资源
if (ioctl(fd, VIDIOC_STREAMOFF, &type) < 0) {
perror("停止捕获失败");
}
munmap(buffer_start, buf.length);
close(fd);
printf("图像已保存为: %s\n", FILENAME);
return 0;
}
Libcamera
定位:基于V4L的更高级框架。
架构层级:高层扩展,提供统一的API和设备管理
目标场景:通用场景(如Linxu桌面应用、APP应用)
依赖关系:需要额外安装libcamera库和相关工具
用法范例
import time
from picamera2 import Picamera2
picam2 = Picamera2()
config = picam2.create_still_configuration(main={"size": (1920, 1080)})
picam2.configure(config)
picam2.start()
#等待相机稳定
time.sleep(2)
picam2.capture_file("image.jpg")
picam2.stop()
常用命令:#查看linux内核uname -rcat /proc/version #查看linux版本详细信息hostnamectl#列出所有文件ls -a#列出带有拥有者的文件ls -l#移动文件mv 源文件或目录 新文件或目录#复制...
一、查看系统安装mysql的情况;rpm -qa | grep -i mysql二、关闭Mysql服务;service mysql stop三、批量删除mysql相关的包;rpm -qa | grep mysql | xargs yum...
fontawesome所有图标class类名单,适用于4.7.0版本。总计有593个图标class类(包含别名):fa-500pxfa-address-bookfa-address-book-ofa-address-cardfa-add...
有同学在群里提问,React有快速应用开发框架,比如nextjs+tailwind.css的技术框架,可以实现网页应用的快速开发。但又听说React开发的网站不太好,那么React到底适合做网站吗?今天丁老师给你解答。其实,目前国内的确...
普通账号个人认证号企业认证号蓝色认证标识薯条推广聚光平台效果广告普通应商业合作以KOL/KOC身份和品牌合作以品牌方身份发起起合作开设店铺个人店铺或个体户(不支持港澳台公司申请)旗舰店店或专卖店(支持港澳台公司申请)主动私信每天10位陌生用...
这几年AI技术的发展已经达到新的高度,从提高个人生产力到企业操作效率的各个方面,AI都扮演着不可或缺的角色。无论是进行语言交流、内容创作、或是任务管理,都有相应的AI工具可以帮助我们更加轻松地完成工作。以下是丁老师觉得不错的AI工具推荐,希...
挺不错的浏览器,强加入了edgeimageviewer图片浏览器,感觉不是那么好,对于开发者来说,更想要原生的东西。包括微信网页版里的表情图片,用edgeimageviewer打开的话,无法保存为gif格式,即使保存了,也只是第一帧静态的。...
丁老师软件,推出图片在线加水印功能,支持图片和PDF文件加水印,可以添加文字水印、图片水印,文字水印支持调整文字大小、文字颜色、位置、透明度等,添加水印后可以自动生成文件,可以下载。在线添加图片水印地址https://www.dls6.co...