(转)c 多张图片生成avi视频

https://www.cnblogs.com/songhe364826110/p/7619949.html

修改了几个参数,可以生成视频了。下载主要是为了学习avi视频格式。最后编一个摄像头生成视频的程序。

本程序把标准avi 视频格式的各种数据结构自定义在文件头(JpegAVI.h)中,所以就不用去下载借用ffmpeg,mplayer等的头文件,库文件了。现在发现,其实编写程序不难,难的是理解定义的各种数据结构和怎样按avi协议去用这些定义的结构。

现在有点明白了为什么学编程要数学好,数学不好就想不出这些数据结构和怎样构建这些文件体系。就象现在虽然有些c的编程知识,可什么也干不了,解决不了实际问题。想创新是不可能的,只有复制别人的成果。

图片文件名为数字,从0开始,不含后缀。

main.c


#include "Jpeg2AVI.h"
#include <string.h>#define JPEG_MAX_SIZE 2000000   //JPEG图像最大字节数,这个数必须大于每一张图片的字节数
#define JPEG_NUM 59  //JPEG图像数量 这个数必须小于等于实际文件数int main()
{FILE *fp_jpg;FILE *fp_avi;int filesize;unsigned char jpg_data[JPEG_MAX_SIZE];      char filename[10];int i = 0;fp_avi = fopen("sample.avi","wb");jpeg2avi_start(fp_avi);for (i = 0; i < JPEG_NUM; i++){memset(filename, 0, 10);memset(jpg_data, 0, JPEG_MAX_SIZE);sprintf(filename, "%d", i);  //int转字符fp_jpg = fopen(filename, "rb");if (fp_jpg != NULL){/*获取JPEG数据大小*/fseek(fp_jpg, 0, SEEK_END);filesize = ftell(fp_jpg);fseek(fp_jpg, 0, SEEK_SET);/*将JPEG数据读到缓冲区*/fread(jpg_data, filesize, 1, fp_jpg);/*将JPEG数据写入AVI文件*/jpeg2avi_add_frame(fp_avi, jpg_data, filesize);}fclose(fp_jpg);}jpeg2avi_end(fp_avi, 1280, 720,1);fclose(fp_avi);printf("end\n");return 0;
}

Jpeg2AVI.h

#ifndef _JPEG2AVI_H_
#define _JPEG2AVI_H_#include <stdio.h>void jpeg2avi_start(FILE *fp);
void jpeg2avi_add_frame(FILE *fp, void *data, unsigned int len);
void jpeg2avi_end(FILE *fp, int width, int height, int fps);typedef struct avi_riff_head
{unsigned char id[4];unsigned int size;unsigned char type[4];
}AVI_RIFF_HEAD, AVI_LIST_HEAD;typedef struct avi_avih_chunk
{unsigned char id[4];            //块ID,固定为avihunsigned int size;              //块大小,等于struct avi_avih_chunk去掉id和size的大小unsigned int us_per_frame;      //视频帧间隔时间(以微秒为单位)unsigned int max_bytes_per_sec; //AVI文件的最大数据率unsigned int padding;           //设为0即可unsigned int flags;             //AVI文件全局属性,如是否含有索引块、音视频数据是否交叉存储等unsigned int total_frames;      //总帧数unsigned int init_frames;       //为交互格式指定初始帧数(非交互格式应该指定为0)unsigned int streams;           //文件包含的流的个数,仅有视频流时为1unsigned int suggest_buff_size; //指定读取本文件建议使用的缓冲区大小,通常为存储一桢图像                                            //以及同步声音所需的数据之和,不指定时设为0unsigned int width;             //视频主窗口宽度(单位:像素)unsigned int height;            //视频主窗口高度(单位:像素)unsigned int reserved[4];       //保留段,设为0即可
}AVI_AVIH_CHUNK;typedef struct avi_rect_frame
{short left;short top;short right;short bottom;
}AVI_RECT_FRAME;typedef struct avi_strh_chunk
{unsigned char id[4];            //块ID,固定为strhunsigned int size;              //块大小,等于struct avi_strh_chunk去掉id和size的大小unsigned char stream_type[4];   //流的类型,vids表示视频流,auds表示音频流unsigned char codec[4];         //指定处理这个流需要的解码器,如JPEGunsigned int flags;             //标记,如是否允许这个流输出、调色板是否变化等,一般设为0即可unsigned short priority;        //流的优先级,视频流设为0即可unsigned short language;        //音频语言代号,视频流设为0即可unsigned int init_frames;       //为交互格式指定初始帧数(非交互格式应该指定为0)unsigned int scale;             //unsigned int rate;              //对于视频流,rate / scale = 帧率fpsunsigned int start;             //对于视频流,设为0即可unsigned int length;            //对于视频流,length即总帧数unsigned int suggest_buff_size; //读取这个流数据建议使用的缓冲区大小unsigned int quality;           //流数据的质量指标unsigned int sample_size;       //音频采样大小,视频流设为0即可AVI_RECT_FRAME rcFrame;         //这个流在视频主窗口中的显示位置,设为{0,0,width,height}即可
}AVI_STRH_CHUNK;/*对于视频流,strf块结构如下*/
typedef struct avi_strf_chunk
{unsigned char id[4];             //块ID,固定为strfunsigned int size;               //块大小,等于struct avi_strf_chunk去掉id和size的大小unsigned int size1;              //size1含义和值同size一样unsigned int width;              //视频主窗口宽度(单位:像素)unsigned int height;             //视频主窗口高度(单位:像素)unsigned short planes;           //始终为1unsigned short bitcount;         //每个像素占的位数,只能是1、4、8、16、24和32中的一个unsigned char compression[4];    //视频流编码格式,如JPEG、MJPG等unsigned int image_size;         //视频图像大小,等于width * height * bitcount / 8unsigned int x_pixels_per_meter; //显示设备的水平分辨率,设为0即可unsigned int y_pixels_per_meter; //显示设备的垂直分辨率,设为0即可unsigned int num_colors;         //含义不清楚,设为0即可unsigned int imp_colors;         //含义不清楚,设为0即可
}AVI_STRF_CHUNK;typedef struct avi_strl_list
{unsigned char id[4];    //块ID,固定为LISTunsigned int size;      //块大小,等于struct avi_strl_list去掉id和size的大小unsigned char type[4];  //块类型,固定为strlAVI_STRH_CHUNK strh;AVI_STRF_CHUNK strf;
}AVI_STRL_LIST;typedef struct avi_hdrl_list
{unsigned char id[4];    //块ID,固定为LISTunsigned int size;      //块大小,等于struct avi_hdrl_list去掉id和size的大小unsigned char type[4];  //块类型,固定为hdrlAVI_AVIH_CHUNK avih;AVI_STRL_LIST  strl;
}AVI_HDRL_LIST;#endif

Jepg2AVI.c

#include "Jpeg2AVI.h"
#include "list.h"
#include <stdlib.h>
#include <string.h>static int nframes;           //总帧数
static int totalsize;         //帧的总大小
static struct list_head list; //保存各帧图像大小的链表,用于写索引块/*链表宿主结构,用于保存真正的图像大小数据*/
struct ListNode
{int value;struct list_head head;
};static void write_index_chunk(FILE *fp)
{unsigned char index[4] = {'i', 'd', 'x', '1'};  //索引块IDunsigned int index_chunk_size = 16 * nframes;   //索引块大小unsigned int offset = 4;struct list_head *slider = NULL;struct list_head *tmpslider = NULL;fwrite(index, 4, 1, fp);fwrite(&index_chunk_size, 4, 1, fp);list_for_each_safe(slider, tmpslider, &list){unsigned char tmp[4] = {'0', '0', 'd', 'c'};  //00dc = 压缩的视频数据unsigned int keyframe = 0x10;                 //0x10表示当前帧为关键帧struct ListNode *node = list_entry(slider, struct ListNode, head);fwrite(tmp, 4, 1, fp);fwrite(&keyframe, 4, 1, fp);fwrite(&offset, 4, 1, fp);fwrite(&node->value, 4, 1, fp);offset = offset + node->value + 8;list_del(slider);free(node);}
}static void back_fill_data(FILE *fp, int width, int height, int fps)
{AVI_RIFF_HEAD riff_head ={{'R', 'I', 'F', 'F'},4 + sizeof(AVI_HDRL_LIST) + sizeof(AVI_LIST_HEAD) + nframes * 8 + totalsize,{'A', 'V', 'I', ' '}};AVI_HDRL_LIST hdrl_list ={{'L', 'I', 'S', 'T'},sizeof(AVI_HDRL_LIST) - 8,{'h', 'd', 'r', 'l'},{{'a', 'v', 'i', 'h'},sizeof(AVI_AVIH_CHUNK) - 8,1000000 / fps, 25000, 0, 0, nframes, 0, 1, 100000, width, height,{0, 0, 0, 0}},{{'L', 'I', 'S', 'T'},sizeof(AVI_STRL_LIST) - 8,{'s', 't', 'r', 'l'},{{'s', 't', 'r', 'h'},sizeof(AVI_STRH_CHUNK) - 8,{'v', 'i', 'd', 's'},{'J', 'P', 'E', 'G'},0, 0, 0, 0, 1, 23, 0, nframes, 100000, 0xFFFFFF, 0,{0, 0, width, height}},{{'s', 't', 'r', 'f'},sizeof(AVI_STRF_CHUNK) - 8,sizeof(AVI_STRF_CHUNK) - 8,width, height, 1, 24,{'J', 'P', 'E', 'G'},width * height * 3, 0, 0, 0, 0}}};AVI_LIST_HEAD movi_list_head ={{'L', 'I', 'S', 'T'},4 + nframes * 8 + totalsize,{'m', 'o', 'v', 'i'}};//定位到文件头,回填各块数据fseek(fp, 0, SEEK_SET);fwrite(&riff_head, sizeof(riff_head), 1, fp);fwrite(&hdrl_list, sizeof(hdrl_list), 1, fp);fwrite(&movi_list_head, sizeof(movi_list_head), 1, fp);
}void jpeg2avi_start(FILE *fp)
{int offset1 = sizeof(AVI_RIFF_HEAD);  //riff head大小int offset2 = sizeof(AVI_HDRL_LIST);  //hdrl list大小int offset3 = sizeof(AVI_LIST_HEAD);  //movi list head大小//AVI文件偏移量设置到movi list head后,从该位置向后依次写入JPEG数据fseek(fp, offset1 + offset2 + offset3, SEEK_SET);//初始化链表list_head_init(&list);nframes = 0;totalsize = 0;
}void jpeg2avi_add_frame(FILE *fp, void *data, unsigned int len)
{unsigned char tmp[4] = {'0', '0', 'd', 'c'};  //00dc = 压缩的视频数据struct ListNode *node = (struct ListNode *)malloc(sizeof(struct ListNode));/*JPEG图像大小4字节对齐*/while (len % 4){len++;}fwrite(tmp, 4, 1, fp);    //写入是否是压缩的视频数据信息fwrite(&len, 4, 1, fp);   //写入4字节对齐后的JPEG图像大小fwrite(data, len, 1, fp); //写入真正的JPEG数据nframes += 1;totalsize += len;/*将4字节对齐后的JPEG图像大小保存在链表中*/if (node != NULL){node->value = len;list_add_tail(&node->head, &list);}
}void jpeg2avi_end(FILE *fp, int width, int height, int fps)
{//写索引块write_index_chunk(fp);//从文件头开始,回填各块数据back_fill_data(fp, width, height, fps);
}

list.h

#ifndef _LIST_H_
#define _LIST_H_struct list_head
{struct list_head *next;struct list_head *prev;
};void list_head_init(struct list_head *list);
void list_add_tail(struct list_head *_new, struct list_head *head);
void list_del(struct list_head *entry);#ifndef offsetof
#define offsetof(TYPE, MEMBER) \
((size_t) &((TYPE *)0)->MEMBER)
#endif#ifndef container_of
#define container_of(ptr, type, member) \
((type *)((char *)ptr - offsetof(type,member)))
#endif/*** list_entry - get the struct for this entry* @ptr:    the &struct list_head pointer.* @type:    the type of the struct this is embedded in.* @member:    the name of the list_struct within the struct.*/
#define list_entry(ptr, type, member) \
container_of(ptr, type, member)/*** list_for_each_safe - iterate over a list safe against removal of list entry* @pos:    the &struct list_head to use as a loop cursor.* @n:        another &struct list_head to use as temporary storage* @head:    the head for your list.*/
#define list_for_each_safe(pos, n, head) \
for (pos = (head)->next, n = pos->next; pos != (head); \
pos = n, n = pos->next)#endif //_LIST_H_

list.c

#include "list.h"
#include <stdio.h>static void __list_add(struct list_head *_new, struct list_head *prev, struct list_head *next)
{next->prev = _new;_new->next = next;_new->prev = prev;prev->next = _new;
}static void __list_del(struct list_head *prev, struct list_head *next)
{next->prev = prev;prev->next = next;
}void list_head_init(struct list_head *list)
{list->next = list;list->prev = list;
}/*** list_add_tail - insert a new entry before the specified head* @_new: new entry to be added* @head: list head to add it before*/
void list_add_tail(struct list_head *_new, struct list_head *head)
{__list_add(_new, head->prev, head);
}/*** list_del - deletes entry from list.* @entry: the element to delete from the list.*/
void list_del(struct list_head *entry)
{__list_del(entry->prev, entry->next);entry->next = NULL;entry->prev = NULL;
}

 

 

 

 

 

 

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/114014.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java面试题-UDP\TCP\HTTP

UDP UDP特性 &#xff08;1&#xff09;UDP是无连接的&#xff1a;发送数据之前不需要像TCP一样建立连接&#xff0c;也不需要释放连接&#xff0c;所以减少了发送和接收数据的开销 &#xff08;2&#xff09;UDP 使用尽最大努力交付&#xff1a;即不保证可靠交付 &#xff0…

Java开发规范记录

不要使用 count(column)或 count(1)来替代 count(*)&#xff0c;count(*)是 SQL92 定义的 标准统计行数的语法&#xff0c;跟数据库无关&#xff0c;跟 NULL 和非 NULL 无关。 注意&#xff1a;count(*)会统计值为 NULL 的行&#xff0c;而 count(列名)不会统计此列为 NULL 值的…

高级深入--day35

反反爬虫相关机制 Some websites implement certain measures to prevent bots from crawling them, with varying degrees of sophistication. Getting around those measures can be difficult and tricky, and may sometimes require special infrastructure. Please consi…

Java面向对象(基础)--package和import关键字的使用

文章目录 一、package关键字的使用1. 说明2. 包的作用3. JDK中主要的包 二、import关键字的使用 一、package关键字的使用 1. 说明 package:包package用于指明该文件中定义的类、接口等结构所在的包。语法格式 举例&#xff1a;pack1\pack2\PackageTest.java package pack1.…

Xshell+screen解决ssh连接 服务器掉线的问题

Linux screen命令解决SSH远程服务器训练代码断开连接后运行中断_linux screen ssh-CSDN博客 Linux命令之screen命令_linux screen_恒悦sunsite的博客-CSDN博客 使用教程&#xff1a; 这里粗略介绍一下 &#xff08;1&#xff09;xshell xftp&#xff08;xshell点这个&#…

Git合并某个分支上的某个提交

1. 首先&#xff0c;确保你当前所在的分支是你要合并分支的父分支。你可以使用以下命令切换到父分支&#xff1a; git checkout <父分支名称> 2. 确保你要合并的分支是可用的。你可以使用以下命令查看所有可用的分支&#xff1a; git branch -a 这将显示所有本地和远程…

组合数(递推版)的初始化

初始考虑为将第一列数和斜对角线上的数进行初始化。 橙色方块由两个绿色方块相加而来&#xff0c;一个为1&#xff0c;一个为0&#xff0c;所以斜对角线都为1&#xff0c;可以通过计算得来&#xff0c;不需要初始化&#xff0c;需要与码蹄集盒子与球 第二类Stirling数&#xf…

Sobel算子详解及例程

Sobel算子是一种经典的边缘检测算子&#xff0c;被广泛应用于图像处理领域。它基于图像亮度的变化率来检测边缘的位置&#xff0c;主要通过计算图像中像素点的梯度来实现。 Sobel算子分为水平和垂直两个方向的算子&#xff0c;记作Gx和Gy。它们分别对图像进行水平和垂直方向的…

Go并发编程之一

一、前言 新年学新语言Go系列文章已经完结&#xff0c;用了最简单的例子去了解Go基础语法&#xff0c;但Go最牛B的是它对并发的友好支持&#xff0c;每一门语言都有它自己独特的优势&#xff0c;如Java适合大型工程化项目&#xff0c;Python适合做数据分析及运维脚本&#xff0…

✔ ★【备战实习(面经+项目+算法)】 10.21学习时间表(总计学习时间:5h30min)(算法刷题:7道)

✔ ★【备战实习&#xff08;面经项目算法&#xff09;】 坚持完成每天必做如何找到好工作1. 科学的学习方法&#xff08;专注&#xff01;效率&#xff01;记忆&#xff01;心流&#xff01;&#xff09;2. 每天认真完成必做项&#xff0c;踏实学习技术 认真完成每天必做&…

工具篇之Axure RP 10的使用

引言 最近在学习原型图&#xff0c;针对画原型图的工具&#xff0c;反复对比墨刀、Axure、xiaopiu后&#xff0c;最终选择Axure。 接下来&#xff0c;我便从Axure RP 10的下载、安装、中文字体、授权等几个方面&#xff0c;来介绍Axure。 一、背景 Axure是一款强大的原型设计…

分布式缓存选型比较:Memcache VS Redis

分布式缓存比较&#xff1a;Memcache VS Redis 1、Redis不仅仅支持简单的k/v类型的数据&#xff0c;同时还提供list&#xff0c;set&#xff0c;zset&#xff0c;hash等数据结构的存储。而memcache只支持简单数据类型&#xff0c;需要客户端自己处理复杂对象 2、Redis支持数据的…

华为OD技术面试-最短距离矩阵(动态规划、广度优先)

背景 记录2023-10-21 晚华为OD三面的手撕代码题&#xff0c;当时没做出来&#xff0c;给面试官说了我的想法&#xff0c;评价&#xff1a;解法复杂了&#xff0c;只是简单的动态规范 或 广度优先算法&#xff0c;事后找资料记录实现方式。 题目 腐烂的橘子 问题描述&#xff…

[SQL | MyBatis] MyBatis 简介

目录 一、MyBatis 简介 1、MyBatis 简介 2、工作流程 二、入门案例 1、准备工作 2、示例 三、Mapper 代理开发 1、问题简介 2、工作流程 3、注意事项 4、测试 四、核心配置文件 mybatis-config.xml 1、environment 2、typeAilases 五、基于 xml 的查询操作 1、…

EtherCAT主站SDO写报文抓包分析

0 工具准备 1.EtherCAT主站 2.EtherCAT从站&#xff08;本文使用步进电机驱动器&#xff09; 3.Wireshark1 抓包分析 1.1 报文总览 本文设置从站1的对象字典&#xff0c;设置对象字典主索引为0x2000&#xff0c;子索引为0x00&#xff0c;设置值为1500。主站通过发送SDO写报文…

STM32-LCD液晶显示

目录 LCD液晶显示 ILI9341液晶控制器简介 液晶屏的信号线和8080时序 使用STM32的FSMC外设模拟8080接口时序 FSMC 功能框图 通讯引脚 存储器控制器 时钟控制逻辑 FSMC的地址映射 FSMC控制异步Nor Flash的时序 FSMC模拟8080时序 FSMC结构体 NOR FLASH时序结构体 F…

由Django-Session配置引发的反序列化安全问题

漏洞成因 漏洞成因位于目标配置文件settings.py下 关于这两个配置项 SESSION_ENGINE&#xff1a; 在Django中&#xff0c;SESSION_ENGINE 是一个设置项&#xff0c;用于指定用于存储和处理会话&#xff08;session&#xff09;数据的引擎。 SESSION_ENGINE 设置项允许您选择不…

Python基础教程:内置函数之字典函数的使用方法

嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! python更多源码/资料/解答/教程等 点击此处跳转文末名片免费获取 len(字典名)&#xff1a; 返回键的个数&#xff0c;即字典的长度 # len(字典名)&#xff1a; # 返回键的个数&#xff0c;即字典的长度dic {a:123,b:456,c:789…

大疆智图(PC):新一代高效率高精度摄影测量软件

大疆智图是一款以二维正射影像与三维模型重建为主的软件&#xff0c;同时提供二维多光谱重建、激光雷达点云处理、精细化巡检等功能。它能够将无人机采集的数据可视化&#xff0c;实时生成高精度、高质量三维模型&#xff0c;满足事故现场、工程监测、电力巡线等场景的展示与精…

UI设计师的主要工作内容优漫动游

一、UI设计师 UI设计师的主要工作内容 1.什么是UI设计师 是现代的一种新兴职业&#xff0c;指从事对软件的人机交互、操作逻辑、界面美观的整体设计工作。 随着科技产品的日益更新和智能手机的普及&#xff0c;社会对软件设计类人才&#xff08;UI设计师&#xff09;提…