首页 > 原创文章 > 技术应用 > 查看文章

如何在Linux上开发Webcam摄像程序?

所属分类:技术应用 来源: 丁老师原创 更新时间:2025-07-25 08:37 浏览: 2319 IP属地: 深圳

有一个需求,需要在Linux客户机上,实现调用摄像头(webcam)来捕捉图像。经过研究,可以使用V4L(Video for Linux2)来实现,一个在Linux下通用的视频驱动框架,支持绝大多数的摄像头(外置和内置摄像头都支持),目前有很多基于linux的网络摄像头和图像采集卡都在使用此框架程序。

如果需要更高级的应用开发,可以选择使用Libcamera,基于V4L开发,是作为V4L的上层扩展,解决了V4L接口的复杂性问题,大幅降低了开发难度,为开发者提供更高级、更建议的摄像头开发框架。

V4L
定位:内核驱动接口(用户空间通过ioctl调用)。
架构层级:底层接口,可以直接操作硬件。
目标场景:高性能、低延迟的应用场景(如工业摄像头、实时处理)
依赖关系:仅需内核支持(大部分Linux系统都已内置)

用法范例

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#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()

相关文章

OpenClaw大龙虾快速安装的方法

最近很多同学都在安装OpenClaw大龙虾,但是在安装时,需要很多客观条件限制,比如需要先连接外网,在安装过程中,还需要自动下载很多相关依赖和环境组件,因网络等原因,安装过程中可能经常会卡死。在此问题的基础上,丁老师特整理出了一套Ope...

VSCode显示函数列表方法

在vscode中,默认是不显示函数列表的,需要手动开启。开启函数列表的具体方法是:1.点击顶部菜单查看,选择打开视图;2.在弹出的选项中,选择大纲;3.这个时候,visual studio code面板的左侧就显示出大纲面板了,面板中显...

OpenMRS系统介绍

OpenMRS(Open Medical Record System)是一个全球性的电子健康记录(EHR)平台,是全球具有影响力的电子医疗记录系统,专注于医疗记录管理,尤其在发展中国家医疗信息化项目中应用广泛,是医疗领域软件的权威代表。...

网站中的CSS/LESS/SCSS都是什么,...

最近有做前端的小伙伴咨询丁老师,网站中使用的不是只有css吗?怎么现在还有Less和Scss,这些都是什么,和CSS有什么区别呢?今天丁老师介绍一下。CSSCSS就是传统的网站样式表,是Cascading Style Sheet的缩写,...

推荐文章

Bagisto-基于Laravel框架的免费...

软件介绍Basgito是基于Laravel框架的免费电子商务平台,可以快速搭建自己企业专属的电子商务交易平台,适用于中小型外贸企业。该电商平台拥有无头laravel商务(headless laravel commerce)的强大功能,可以快...

丁老师软件推出图片在线加水印功能

丁老师软件,推出图片在线加水印功能,支持图片和PDF文件加水印,可以添加文字水印、图片水印,文字水印支持调整文字大小、文字颜色、位置、透明度等,添加水印后可以自动生成文件,可以下载。在线添加图片水印地址https://www.dls6.co...

二维码可以存储多少个信息?

二维码码有三个参数: 数据类型、大小(“像素 ”数)和纠错级别。能存储多少信息也取决于这些参数。例如,纠错级别越低,可存储的信息越多,但代码越难被读者识别。最大尺寸和最小纠错值如下:最大字符数 7089个字符(纯数字)字母数字 4296个字...

AI工具大全:好用的AI工具推荐(持续更新中...

这几年AI技术的发展已经达到新的高度,从提高个人生产力到企业操作效率的各个方面,AI都扮演着不可或缺的角色。无论是进行语言交流、内容创作、或是任务管理,都有相应的AI工具可以帮助我们更加轻松地完成工作。以下是丁老师觉得不错的AI工具推荐,希...

当前在线
免费咨询