python:mido 提取 midi文件中某一音轨的音乐数据

news2025/4/24 11:56:38

pip install mido

使用 mido库可以方便地处理 MIDI 文件,提取其中音轨的音乐数据。

1.下面的程序会读取指定的 MIDI 文件,并提取指定编号音轨的音乐数据,主要包括音符事件等信息。

编写 mido_extract.py  如下

# -*- coding: utf-8 -*-
""" 用 mido 提取 midi文件中某一音轨的音乐数据 """
import os
import sys
from tkinter import filedialog
import mido

def extract_track_data(midi_file, track_no):
    try:
        track_data = []
        tracks = midi_file.tracks
        for i,track in enumerate(tracks):
            print(f"Track {i}: {track.name}")
        if track_no < len(tracks):
            track = tracks[track_no]
            for msg in track:
                if msg.type in ['note_on', 'note_off', 'control_change', 'program_change']:
                    track_data.append(msg)            
        else:
            print(f"音轨编号 {track_no} 超出范围,该MIDI文件共有 {len(tracks)}条音轨。")
        return track_data
    except:
        print(" 无法读取MIDI文件.")
        return None


def write_track_to_midi(track_data, output_file_path, original_mid):
    new_mid = mido.MidiFile(ticks_per_beat=original_mid.ticks_per_beat)
    new_track = mido.MidiTrack()
    for msg in track_data:
        new_track.append(msg)
    new_mid.tracks.append(new_track)
    new_mid.save(output_file_path)


# main()
if len(sys.argv) ==1:
    filetypes = [('mid file','.mid')]
    f1 = filedialog.askopenfilename(initialdir='D:/Music', filetypes=filetypes)
elif len(sys.argv) ==2:
    f1 = sys.argv[1]
else:
    print('usage: python mido_extract.py file1.mid')
    sys.exit(1)

if not os.path.exists(f1):
    print(f"{f1} is not exists.")
    sys.exit(2)

fn,ext = os.path.splitext(f1)
if ext.lower() != '.mid':
    print('ext is not .mid ')
    sys.exit(2)

midi_file_path = f1  # 替换为实际的MIDI文件路径
x = 1  # 替换为想要提取的音轨编号,从0开始
output_file_path = f"track_{x}.mid"  # 替换为输出的MIDI文件路径
try:
    original_mid = mido.MidiFile(midi_file_path)
    data = extract_track_data(original_mid, x)
    if data:
        write_track_to_midi(data, output_file_path, original_mid)
        print(f"已成功将track {x} 数据写入 {output_file_path}")
except mido.MidiFileError:
    print(f"无法读取MIDI文件 {midi_file_path}")

运行 python mido_extract.py moon_river.mid
生成 track_1.mid

上述代码中,extract_track_data函数负责提取指定音轨的数据,write_track_to_midi函数则将提取的数据写入新的 MIDI 文件。你需要把midi_file_path换成原始 MIDI 文件路径,track_number设为目标音轨编号,output_file_path指定输出文件路径。


2.下面的代码会复制保留原 MIDI 文件中 Track 0 的全部信息,同时提取 Track 1 的音乐数据,然后将它们合并到一个新的 MIDI 文件中。

编写 mido_extract1.py  如下

# -*- coding: utf-8 -*-
""" 用 mido 复制保留 Track 0 全部信息,提取 Track 1 音乐数据 """
import mido

def copy_track0_and_extract_track1(midi_file_path, output_file_path):
    try:
        # 读取原 MIDI 文件
        original_mid = mido.MidiFile(midi_file_path)

        if len(original_mid.tracks) < 2:
            print("MIDI 文件中没有足够的音轨(至少需要 2 条音轨)。")
            return

        # 创建新的 MIDI 文件,复制原文件的 ticks_per_beat
        new_mid = mido.MidiFile(ticks_per_beat=original_mid.ticks_per_beat)

        # 复制 Track 0 到新的 MIDI 文件
        new_mid.tracks.append(original_mid.tracks[0])

        # 提取 Track 1 的音乐数据
        track1 = original_mid.tracks[1]
        track1_data = []
        for msg in track1:
            if msg.type in ['note_on', 'note_off', 'control_change', 'program_change']:
                track1_data.append(msg)

        # 创建新的音轨并添加 Track 1 的数据
        new_track1 = mido.MidiTrack()
        for msg in track1_data:
            new_track1.append(msg)
        new_mid.tracks.append(new_track1)

        # 保存新的 MIDI 文件
        new_mid.save(output_file_path)
        print(f"已成功将 Track 0 信息和 Track 1 数据写入 {output_file_path}")

    except mido.MidiFileError:
        print(f"无法读取 MIDI 文件 {midi_file_path}")


if __name__ == "__main__":
    midi_file_path = "your_midi_file.mid"  # 替换为实际的 MIDI 文件路径
    output_file_path = "new_midi_file.mid"  # 替换为输出的 MIDI 文件路径
    copy_track0_and_extract_track1(midi_file_path, output_file_path)
    

代码说明:

  1. 读取原 MIDI 文件:使用 mido.MidiFile 读取指定路径的 MIDI 文件。
  2. 检查音轨数量:确保 MIDI 文件中至少有 2 条音轨。
  3. 创建新的 MIDI 文件:使用原文件的 ticks_per_beat 创建新的 MIDI 文件。
  4. 复制 Track 0:将原文件的 Track 0 直接添加到新的 MIDI 文件中。
  5. 提取 Track 1 数据:遍历 Track 1,提取 note_onnote_offcontrol_change 和 program_change 类型的消息。
  6. 创建新音轨并添加数据:创建一个新的音轨,并将提取的 Track 1 数据添加到该音轨中,然后将该音轨添加到新的 MIDI 文件中。
  7. 保存新的 MIDI 文件:将新的 MIDI 文件保存到指定的输出路径。

你需要将 midi_file_path 替换为实际的 MIDI 文件路径,将 output_file_path 替换为你希望保存的新 MIDI 文件的路径。

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

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

相关文章

URP-UGUI交互功能实现

一、非代码层面实现交互&#xff08;SetActive&#xff09; Button &#xff1a;在OnClick&#xff08;&#xff09;中添加SetActive方法&#xff08;但是此时只首次有效&#xff09; Toggle &#xff1a;在OnClick&#xff08;&#xff09;中添加动态的SetActive方法 &#…

UniGoal 具身导航 | 通用零样本目标导航 CVPR 2025

UniGoal的提出了一个通用的零样本目标导航框架&#xff0c;能够统一处理多种类型的导航任务 &#xff08;如对象类别导航、实例图像目标导航和文本目标导航&#xff09;&#xff0c;而无需针对特定任务进行训练或微调。 它的特点是 图匹配与多阶段探索策略&#xff01;&#x…

通过Quartus II实现Nios II编程

目录 一、认识Nios II二、使用Quartus II 18.0Lite搭建Nios II硬件部分三、软件部分四、运行项目 一、认识Nios II Nios II软核处理器简介 Nios II是Altera公司推出的一款32位RISC嵌入式处理器&#xff0c;专门设计用于在FPGA上运行。作为软核处理器&#xff0c;Nios II可以通…

Linux/AndroidOS中进程间的通信线程间的同步 - IPC方式简介

前言 从来没有总结过Linux/Android系统中进程间的通信方式和线程间的同步方式&#xff0c;这个专栏就系统总结讨论一下。首先从标题可知&#xff0c;讨论问题的主体是进程和线程、通信和同步&#xff1b;在这里默认你理解进程和线程的区别。通信和同步有什么概念上的区别&…

Windows:注册表配置应用

0、简介 本篇博客记录一下&#xff0c;日常的系统注册表配置选项&#xff0c;以防再次遇到问题不知如何解决。 1、开机启动配置 HKEY_LOCAL_MACHINE\Software\Microsoft\Windows\CurrentVersion\Run :: 此位置存储了所有用户登录时需要启动的程序。 在该项下新建字符串值&#…

WebXR教学 05 项目3 太空飞船小游戏

准备工作 自动创建 package.json 文件 npm init -y 安装Three.js 3D 图形库&#xff0c;安装现代前端构建工具Vite&#xff08;用于开发/打包&#xff09; npm install three vite 启动 Vite 开发服务器&#xff08;推荐&#xff09;&#xff08;正式项目开发&#xff09; …

达梦统计信息收集情况检查

查询达梦某个对象上是否有统计信息 select id,T_TOTAL,N_SMAPLE,N_DISTINCT,N_NULL,BLEVEL,N_LEAF_PAGES,N_LEAF_USED_PAGES,LAST_GATHERED from sysstats where id IN (select id from sysobjects where upper(name)upper(&objname));可能有系统对象&#xff0c;可以增加…

【matlab】气泡图的应用

【matlab】气泡图的应用 .rtcContent { padding: 30px; } .lineNode {font-size: 12pt; font-family: "Times New Roman", Menlo, Monaco, Consolas, "Courier New", monospace; font-style: normal; font-weight: normal; } clear load zb_equi.mat load …

飞帆控件:在编辑模式下额外加载的库

飞帆是一个自由的控件设计平台。在飞帆中&#xff0c;我们可以很方便地创建基于 Vue 2 组件的控件&#xff0c;并使用控件来搭建网页。 他山之石&#xff0c;可以攻玉。在创建控件中&#xff0c;使用 js 、css 依赖库能让我们的控件更强大。 有些时候&#xff0c;在编辑模式下…

Super-Vlan和MUX-Vlan的原理、配置、区别

Super-Vlan 原理 Super-Vlan也叫Aggregate-Vlan。 一般的三层交换机中&#xff0c;通常是采用一个VLAN对应一个vlanif接口的方式实现广播域之间的互通&#xff0c;这在某些情况下导致了IP地址的浪费。因为一个VLAN对应的子网中&#xff0c;子网号、子网定向广播地址、子网缺…

el-table怎么显示 特殊单元格的值

1. 在 el-table-column 上绑定了 formatter 方法 formatEntityName &#xff0c;它会对每一行该列的数据&#xff08; cellValue &#xff09;进行处理。 2. 在 formatEntityName 方法中&#xff0c;尝试对传入的 cellValue 进行 JSON.parse 操作&#xff0c;并根…

2025-04-23 Python深度学习3——Tensor

文章目录 1 张量1.1 数学定义1.2 PyTorch中的张量 2 创建 Tensor2.1 直接创建**torch.tensor()****torch.from_numpy()** 2.2 依据数值创建**torch.zeros() / torch.zeros_like()****torch.ones() / torch.ones_like()****torch.full() / torch.full_like()****torch.arange() …

在统信UOS/麒麟Kylin OS操作系统中配置APT和GIT代理

在统信UOS/麒麟Kylin OS操作系统中配置APT和GIT代理 在内网环境中&#xff0c;直接访问外部资源可能会受到限制&#xff0c;这时候配置APT和GIT的代理就显得尤为重要。本文将详细介绍如何在统信UOS和麒麟Kylin OS操作系统中配置APT和GIT的代理。 为什么需要配置APT和GIT代理&…

第十七讲、Isaaclab中使用操作空间控制器

0 前言 官方教程&#xff1a;https://isaac-sim.github.io/IsaacLab/main/source/tutorials/05_controllers/run_osc.html IsaacsimIsaaclab安装&#xff1a;https://blog.csdn.net/m0_47719040/article/details/146389391?spm1001.2014.3001.5502 有时候&#xff0c;仅使用…

基于SpringBoot的校园二手商品在线交易系统+含项目运行说明文档

基于SpringBoot的校园二手商品在线交易系统含项目运行说明文档 专注校园二手交易平台是一个基于Java的在线市场&#xff0c;专为学生设计&#xff0c;便于买卖二手商品。平台提供全面的用户管理功能&#xff0c;包括学生、管理员和二手商品卖家账户管理。商品管理功能允许用户…

详解springcloud gateway工作原理、断言、filter、uri、id、全局跨域、globalfilter等以及关键源码实现

1.gateway概念 网关就是当前微服务项目的"统一入口"程序中的网关就是当前微服务项目对外界开放的统一入口所有外界的请求都需要先经过网关才能访问到我们的程序提供了统一入口之后,方便对所有请求进行统一的检查和管理 2. 网关的主要功能 将所有请求统一经过网关网…

C++面向对象特性之继承篇

C语音是面向过程的语言&#xff0c;而C在其之上多了面向对象的特性&#xff0c;面向对象三大特性:封装性、继承性、多态性。今天主包来讲讲自己学到的关于C继承特性的知识。 一、继承是什么 继承是提高代码复用的一种重要手段。正如C的模版、泛型编程等等都是为了实现代码复用…

【AI News | 20250423】每日AI进展

AI Repos 1、suna Suna是一款完全开源的AI助手&#xff0c;旨在通过自然对话帮助用户轻松完成现实世界的任务。它作为您的数字伙伴&#xff0c;提供研究、数据分析和日常问题解决等功能&#xff0c;并结合强大的能力与直观的界面&#xff0c;理解您的需求并交付成果。Suna的工…

【学习准备】算法和开发知识大纲

1 缘起 今年&#xff08;2025年&#xff09;的职业升级结果&#xff1a;不通过。没办法升职加薪了。 需要开始完善学习&#xff0c;以应对不同的发展趋势&#xff0c;为了督促自己学习&#xff0c;梳理出相关学习大纲。 分为算法和开发两部分。 算法&#xff0c;包括基础算法和…

第七篇:linux之基本权限、进程管理、系统服务

第七篇&#xff1a;linux之基本权限、进程管理、系统服务 文章目录 第七篇&#xff1a;linux之基本权限、进程管理、系统服务一、基本权限1、什么是权限&#xff1f;2、为什么要有权限&#xff1f;3、权限与用户之间的关系&#xff1f;4、权限对应的数字含义5、使用chmod设定权…