C语言 读取 MIDI文件头部
在C语言中直接读取MIDI文件并不简单,因为MIDI文件是一种包含音乐事件(如音符的开始和结束、控制信号等)的二进制格式,而不是像文本文件那样容易解析。不过,你可以通过以下步骤来实现:
- 了解MIDI文件格式:
- MIDI文件有多种格式,最常见的是SMF (Standard MIDI File) 格式,也被称为Type 0、Type 1或Type 2。
- 你需要了解MIDI文件的各个部分,包括文件头、轨道头和轨道事件等。
- 编写解析器:
- 编写一个C程序来读取MIDI文件的二进制内容。
- 首先,你需要读取文件头以获取文件的整体信息(如文件类型、轨道数等)。
- 然后,逐个读取轨道头,了解每个轨道的详细信息。
- 接下来,解析轨道中的MIDI事件,这通常涉及解析时间戳和事件数据。
- 处理MIDI事件:
- 对于每个MIDI事件,你需要解析事件数据以了解它是什么类型的事件(如音符开始、音符结束、控制更改等)。
- 根据事件类型,你可以执行相应的操作,如播放音符、更改音量或发送其他MIDI消息。
- 使用库:
- 考虑到MIDI文件的复杂性,你可能希望使用现有的库来帮助解析和处理MIDI文件。
- 例如,你可以查找支持C语言的MIDI库,如
libsmf或libmidi(请注意,这些库可能不存在或已过时,因为MIDI文件处理在C语言中并不常见)。
- 集成到项目中:
- 一旦你能够解析和处理MIDI文件,你可以将其集成到你的C项目中。
- 这可能涉及将MIDI事件转换为音频输出(如使用MIDI输出设备或软件合成器)或将MIDI数据转换为其他格式(如MIDI转乐谱)。
- 测试和调试:
- 编写测试用例来验证你的MIDI解析器是否能够正确处理各种MIDI文件。
- 使用调试工具来查找和修复任何错误或问题。
请注意,直接处理MIDI文件需要深入了解MIDI规范和相关文件格式。如果你不熟悉这些概念,可能需要花费一些时间来学习它们。此外,由于MIDI文件的复杂性,编写一个健壮且可靠的MIDI解析器可能是一个相当复杂的任务。
如果你只是想在C语言项目中播放MIDI文件,而不是解析和处理MIDI文件的内部细节,那么一个更简单的方法是使用外部MIDI播放器或库来播放MIDI文件,并通过命令行或API接口与你的C程序进行交互。
请先看百度百科:MIDI文件格式
编写 mid_head.c 读取 midi 文件头部:
#include <stdio.h>
#include <stdint.h>typedef struct {char ctag[4]; // chunk_tag: MThdint32_t chunk_size;// 指定Midi的格式: 00 00单音轨; 00 01多音轨,且同步; 00 02多音轨,但不同步uint16_t geshi; uint16_t tracks; // 轨道数:=实际音轨数字 +1个全局音轨// 指定基本时间格式类型: 类型1:定义一个四分音符的tick数; // 类型2:定义每秒中SMTPE帧的数量及每个SMTPE帧的ticksuint16_t ticks; char ttag[4]; // track_tag: MTrkuint8_t t_id;uint16_t track_size;
} MidiHeader;uint16_t swapUint16(uint16_t shortValue){return ((shortValue & 0x00FF ) <<8) | ((shortValue & 0xFF00)>>8);
}int32_t swapInt32(int32_t intValue){int32_t temp = 0;temp = ((intValue & 0x000000FF) <<24) +((intValue & 0x0000FF00) <<8) +((intValue & 0x00FF0000) >>8) +((intValue & 0xFF000000) >>24);return temp;
}int main(int argc, char *argv[])
{if (argc < 2) {printf("Usage: %s <filename>\n", argv[0]);return 1;}const char *f1 = argv[1]; // filenameFILE *file = fopen(f1, "rb");if (!file) {perror("Error opening file");return -1;}MidiHeader hd;if (fread(&hd, sizeof(MidiHeader), 1, file) != 1) {fclose(file);perror("Error reading file head");return -1;}// 打印读取到的数据,验证读取成功printf("Chunk tag: %s\n", hd.ctag);printf("Chunk Size: %04d\n", swapInt32(hd.chunk_size));printf("geshi:%d, tracks:%d, ticks:%d\n",swapUint16(hd.geshi),swapUint16(hd.tracks),swapUint16(hd.ticks));printf("Track tag: %s\n", hd.ttag);printf("track id: %x, track Size: %d\n", hd.t_id, swapUint16(hd.track_size));if (fseek(file, swapUint16(hd.track_size)-2, 1) !=0) {fclose(file);perror("Error fseek file ");return -1;}char t1tag[5];if (fread(&t1tag, sizeof(char), 4, file) != 4) {fclose(file);perror("Error reading file head");return -1;}printf("track1 tag: %s\n", t1tag);uint32_t track1_size;if (fread(&track1_size, sizeof(uint32_t), 1, file) != 1) {fclose(file);perror("Error reading file head");return -1;}printf("track1 size: %d\n", swapInt32(track1_size)); fclose(file);return 0;
}
where gcc
D:\Strawberry\c\bin\gcc.exe
编译 gcc mid_head.c -o mid_head.exe
运行 mid_head happy_birthday.mid
mid_head happy_birthday.mid
Chunk tag: MThd
Chunk Size: 0006
geshi:1, tracks:2, ticks:1024
Track tag: MTrk
Track id: 0, Track Size: 20
Track1 tag: MTrk
track1 size: 247
为了对单个几十MB的.mid 文件采样数据,读取.mid 文件头部 4080 bytes
Unix 命令 head -c 4080 sample1.mid > temp1.mid
运行 strings temp1.mid
相关文章:
C语言 读取 MIDI文件头部
在C语言中直接读取MIDI文件并不简单,因为MIDI文件是一种包含音乐事件(如音符的开始和结束、控制信号等)的二进制格式,而不是像文本文件那样容易解析。不过,你可以通过以下步骤来实现: 了解MIDI文件格式&am…...
C# Winform实现五子棋游戏(代完善)
实现了基本的玩法。 BoardController.cs using System;namespace GomokuGame {public class BoardController{private static BoardController instance;private readonly int[,] board;private const int boardSize 15;private BoardController(){board new int[boardSize…...
文档档案管理系统整体建设方案书(实际项目原件word2024)
1.系统概述 1.1.需求描述 1.2.需求分析 1.3.重难点分析 1.4.重难点解决措施 2.系统架构设计 2.1.系统架构图 2.2.关键技术 数据备份技术 3.系统功能设计 3.1.功能清单列表 3.2.基础数据管理 3.3.位置管理 3.4.文档使用 3.5.文档管理 软件全套资料包获取方式①:软件项…...
React与Vue的区别?
一、区别: 1. 语法 Vue采用自己特有的模板语法; React是单向的,采用jsx语法创建react元素。 2.监听数据变化的实现原理不同 Vue2.0 通过Object.defineproperty()方法的getter/setter属性, 实现数据劫持, 每次修改完数据会触发diff算法(双端对比) …...
leetcode 2115.从给定原材料中找到所有可以做出的菜
思路:拓补排序,哈希表 在思路上其实很好发现,我们需要有一个明确的做菜顺序,也就是说需要定下来我们根据原材料先做哪些菜,然后做完该做的菜之后,后来又能做哪些菜。 你也发现了,这个顺序其实…...
Opencompass模型评测教程
模型评测 模型评测非常关键,目前主流的方法主要可以概括为主观评测和客观评测,主观评测又可以分为两种形式:人工判断或者和模型竞技场。客观评测一般采用评测数据集的形式进行模型评测。本教程使用Opencompass工具进行对Internlm2-7b模型进行…...
什么是安全测试,如何进行安全测试?
什么是安全测试? 概述 安全测试是一种旨在识别系统、网络或应用程序中的安全漏洞的测试方法。其目标是确保系统能够抵御恶意攻击,保护数据的机密性、完整性和可用性。安全测试通常包括漏洞扫描、渗透测试、代码审计和安全评估等多个方面。 安全测试的…...
ros的pcl库中对于自己定义的消息,调用pcl库时总是报错 c++
首先定义自己的消息类型 struct CustomPoint { // 定义点类型结构PCL_ADD_POINT4D; // 该点类型有4个元素float intensity 0.0;uint32_t zone;uint32_t ring;uint32_t sector;EIGEN_MAKE_ALIGNED_OPERATOR_NEW // 确保new操作符对齐操作 } EIGEN_ALIGN16; // 强制SSE对齐POIN…...
DataFrame—数据汇总6
文章最前: 我是Octopus,这个名字来源于我的中文名--章鱼;我热爱编程、热爱算法、热爱开源。所有源码在我的个人github ;这博客是记录我学习的点点滴滴,如果您对 Python、Java、AI、算法有兴趣,可以关注我的…...
Java入门基础学习笔记41——实体类
实体JavaBean/实体类: 就是一种特殊形式的类。 1)这个类中的成员变量都要私有,并且要对外提供相应的getXXX,setXXX的方法。 2)类中必须要有一个公共的无参的构造器。其他的构造器可写可不写。 右键菜单中࿰…...
【Linux】信号之信号的保存和处理详解
🤖个人主页:晚风相伴-CSDN博客 💖如果觉得内容对你有帮助的话,还请给博主一键三连(点赞💜、收藏🧡、关注💚)吧 🙏如果内容有误或者有写的不好的地方的话&…...
基于Django的图书管理系统
文章目录 前言一、页面展示1.登录2.前端页面3.后端页面 二、项目上传(1)导入数据库(2)导入项目(3)数据库密码修改(4)进入网站 总结 前言 本网站调用Django编写了图书管理网站&#…...
js实现元素根据鼠标滚轮滚动向左右上下滑动着从模糊到清楚显示出来
html代码 <div ref{test} id"animatedElement" className"not-animated"> <div style{{width:"100px",height:"50px",backgroundColor:"red"}}> </div> </div> JS代码 const te…...
yocto学习
bitbake命令单独编译u-boot: $ bitbake -c compile -f u-boot-imx $ bitbake -c deploy -f u-boot-imx //部署编译生成的u-boot镜像到deploy bitbake命令单独编译kernel: bitbake -c compile -f linux-imx //编译内核 bitbake -c deploy -f linux-imx /…...
【IC设计】牛客网-序列检测习题总结
文章目录 状态机基础知识VL25 输入序列连续的序列检测VL26 含有无关项的序列检测VL27 不重叠序列检测VL28 输入序列不连续的序列检测参考资料 状态机基础知识 VL25 输入序列连续的序列检测 timescale 1ns/1ns module sequence_detect(input clk,input rst_n,input a,output re…...
python爬虫登录到海康相机管理页面
简述 1.最近接到个任务是在管理页面更改相机的某个参数,下载官方的sdk貌似没有提供这个接口,所以只能自己写爬虫登录发请求了。 1.主要步骤 1.1 发送get请求获取到salt,sessionID,challenge等信息 http://admin:123456192.168.…...
9.Docker网络
文章目录 1、Docker网络简介2、常用基本命令3、网络模式对比举例3.1、bridge模式3.2、host模式3.3、none模式3.4、container模式3.5、自定义网络 1、Docker网络简介 作用: 容器间的互联和通信以及端口映射容器IP变动时候可以通过服务名直接进行网络通信而不受到影…...
Windows VS2022 C语言使用 sqlite3.dll 访问 SQLite数据库
今天接到一个学生C语言访问SQLite数据库的的需求: 第一步,SQLite Download Page下载 sqlite3.dll 库 下载解压,发现只有两个文件: 于是使用x64 Native Tools Command Prompt 终端 生成 sqlite3.lib 和 sqlite3.exp文件 LIB -def:sqlite3.def -out:sqlite3.lib -machin…...
java库和包的概念
在Java中,"库"和"包"是两个不同的概念,但它们之间存在着密切的关联。 库(Library) 定义:库是一组已经编写好的代码和资源,用于解决特定的问题或提供特定的功能。它可以包含一个或多个…...
mysql内存结构
一:逻辑存储结构:表空间->段->区->页->行、 表空间:一个mysql实例对应多个表空间,用于存储记录,索引等数据。 段:分为数据段,索引段,回滚段。innoDB是索引组织表&…...
【kafka】Golang实现分布式Masscan任务调度系统
要求: 输出两个程序,一个命令行程序(命令行参数用flag)和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽,然后将消息推送到kafka里面。 服务端程序: 从kafka消费者接收…...
云计算——弹性云计算器(ECS)
弹性云服务器:ECS 概述 云计算重构了ICT系统,云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台,包含如下主要概念。 ECS(Elastic Cloud Server):即弹性云服务器,是云计算…...
oracle与MySQL数据库之间数据同步的技术要点
Oracle与MySQL数据库之间的数据同步是一个涉及多个技术要点的复杂任务。由于Oracle和MySQL的架构差异,它们的数据同步要求既要保持数据的准确性和一致性,又要处理好性能问题。以下是一些主要的技术要点: 数据结构差异 数据类型差异ÿ…...
PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...
【服务器压力测试】本地PC电脑作为服务器运行时出现卡顿和资源紧张(Windows/Linux)
要让本地PC电脑作为服务器运行时出现卡顿和资源紧张的情况,可以通过以下几种方式模拟或触发: 1. 增加CPU负载 运行大量计算密集型任务,例如: 使用多线程循环执行复杂计算(如数学运算、加密解密等)。运行图…...
【python异步多线程】异步多线程爬虫代码示例
claude生成的python多线程、异步代码示例,模拟20个网页的爬取,每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程:允许程序同时执行多个任务,提高IO密集型任务(如网络请求)的效率…...
算法笔记2
1.字符串拼接最好用StringBuilder,不用String 2.创建List<>类型的数组并创建内存 List arr[] new ArrayList[26]; Arrays.setAll(arr, i -> new ArrayList<>()); 3.去掉首尾空格...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...
Git常用命令完全指南:从入门到精通
Git常用命令完全指南:从入门到精通 一、基础配置命令 1. 用户信息配置 # 设置全局用户名 git config --global user.name "你的名字"# 设置全局邮箱 git config --global user.email "你的邮箱example.com"# 查看所有配置 git config --list…...
深入理解Optional:处理空指针异常
1. 使用Optional处理可能为空的集合 在Java开发中,集合判空是一个常见但容易出错的场景。传统方式虽然可行,但存在一些潜在问题: // 传统判空方式 if (!CollectionUtils.isEmpty(userInfoList)) {for (UserInfo userInfo : userInfoList) {…...
