用Python代码写首《本草纲目》

简介: 作为刘畊宏女孩,咱必须要支棱起来!!也没啥其他技能,就用Python写首《本草纲目》给刘教练应援一下吧!没错,Python不仅可以做数据分析,还可以用来写歌。绝绝子!

用的是Python的mido库,先来测试几个音。


#导入mido库
from mido import Message, MidiFile, MidiTrack   
#创建.mid文件
mid = MidiFile()       
#创建一个音轨
track = MidiTrack()    
#设置音色,1表示明亮的钢琴
track.append(Message('program_change',program=1, time=0))
#播放C4
track.append(Message('note_on', note=60,velocity=64, time=480))
track.append(Message('note_off', note=60,velocity=127, time=480))
#播放D4
track.append(Message('note_on', note=62,velocity=64, time=480))
track.append(Message('note_off', note=62,velocity=127, time=480))
midi_file="test.mid"
#保存.mid文件
mid.save(midi_file)


运行后生成mid文件,Mac电脑可以用库乐队播放,Windows电脑可以用MidiEditor播放。


测试没问题,咱们正式开始写歌~


第一步:


找到《本草纲目》简谱


image.png

第二步:


简谱翻译成mido库能看懂的数据


def play_note(note, length, track, base_num=1, delay=0, velocity=1.0, channel=0):
    meta_time = 60 * 60 * 10 / bpm
    major_notes = [0, 2, 2, 1, 2, 2, 2, 1]
    base_note = 60
    track.append(Message("program_change", program=2, time=0))
    track.append(
        Message('note_on', note=base_note + base_num * 12 + sum(major_notes[0:note]), velocity=round(64 * velocity),
                time=round(delay * meta_time), channel=channel))
    track.append(
        Message('note_off', note=base_note + base_num * 12 + sum(major_notes[0:note]), velocity=round(64 * velocity),
                time=round(meta_time * length), channel=channel))


第三步:


创建音轨,生成mid文件


fileName = "bcgm.mid"
mid = MidiFile()  # 创建midi文件
pg.mixer.init()
# 创建音轨
track1 = play_note(2,mids1,0)
track2 = play_note(44,mids2,4)
mid.tracks.append(track1)
mid.tracks.append(track2)
mid.save(fileName)


最后播放音乐,听听最终效果吧~


image.png


原视频点击网址


https://blog.csdn.net/shine_a/article/details/124407036?spm=1001.2014.3001.5502


用Python写首《本草纲目》


完整代码:


# 本草纲目
from mido import Message, MidiFile, MidiTrack
import pygame as pg
'''
播放一个音符
note 简谱的音符,21到108,以半音作为最小单位
length 音符长度
Program 确定乐器 2钢琴
velocity 音量
time 间隔时间
'''
def play_note(program,mids,channel):
    track = MidiTrack()
    track.append(Message("program_change", program=program, time=0))
    for arrs in mids:
        if arrs[0] == 0:
            track.append(Message('note_on', note=arrs[1], velocity=arrs[2], time=arrs[3], channel=channel))
        else:
            track.append(Message('note_off', note=arrs[1], velocity=arrs[2], time=arrs[3], channel=channel))
    return track
# 定义音符 arr[0] 0代表note_on,1代表note_off  arr[1]=note  arr[2]=velocity arr[3]=time
mids1 = [[0, 75, 80, 48], [1, 75, 64, 216], [0, 70, 80, 24], [1, 70, 64, 108],
        [0, 73, 80, 12], [1, 73, 64, 108], [0, 75, 80, 12], [1, 75, 64, 216], [0, 75, 80, 24],
        [1, 75, 64, 216], [0, 70, 80, 24], [1, 70, 64, 108], [0, 73, 80, 12], [1, 73, 64, 108],
        [0, 75, 80, 12], [1, 75, 64, 216], [0, 75, 80, 24], [1, 75, 64, 432], [0, 75, 80, 48],
        [1, 75, 64, 216], [0, 70, 80, 24], [1, 70, 64, 108], [0, 73, 80, 12], [1, 73, 64, 108],
        [0, 75, 80, 12], [1, 75, 64, 216], [0, 75, 80, 24], [1, 75, 64, 216], [0, 70, 80, 24],
        [1, 70, 64, 108], [0, 73, 80, 12], [1, 73, 64, 108], [0, 70, 80, 12], [1, 70, 64, 216],
        [0, 63, 80, 24], [1, 63, 64, 432], [0, 75, 80, 48], [1, 75, 64, 216], [0, 70, 80, 24],
        [1, 70, 64, 108], [0, 73, 80, 12], [1, 73, 64, 108], [0, 75, 80, 12], [1, 75, 64, 216],
        [0, 75, 80, 24], [1, 75, 64, 216], [0, 70, 80, 24], [1, 70, 64, 108], [0, 73, 80, 12],
        [1, 73, 64, 108], [0, 75, 80, 12], [1, 75, 64, 216], [0, 75, 80, 24], [1, 75, 64, 432],
        [0, 75, 80, 48], [1, 75, 64, 216], [0, 70, 80, 24], [1, 70, 64, 108], [0, 73, 80, 12],
        [1, 73, 64, 108], [0, 75, 80, 12], [1, 75, 64, 216], [0, 75, 80, 24], [1, 75, 64, 216],
        [0, 70, 80, 24], [1, 70, 64, 108], [0, 73, 80, 12], [1, 73, 64, 108], [0, 75, 80, 12],
        [1, 75, 64, 216], [0, 75, 80, 24], [1, 75, 64, 432], [0, 76, 80, 0], [1, 76, 64, 216],
        [0, 71, 80, 24], [1, 71, 64, 108], [0, 74, 80, 12], [1, 74, 64, 108], [0, 76, 80, 12],
        [1, 76, 64, 216], [0, 76, 80, 24], [1, 76, 64, 216], [0, 71, 80, 24], [1, 71, 64, 108],
        [0, 74, 80, 12], [1, 74, 64, 108], [0, 76, 80, 12], [1, 76, 64, 216], [0, 76, 80, 24],
        [1, 76, 64, 432], [0, 76, 80, 48], [1, 76, 64, 216], [0, 71, 80, 24], [1, 71, 64, 108],
        [0, 74, 80, 12], [1, 74, 64, 108], [0, 76, 80, 12], [1, 76, 64, 216], [0, 76, 80, 24],
        [1, 76, 64, 216], [0, 71, 80, 24], [1, 71, 64, 108], [0, 74, 80, 12], [1, 74, 64, 108],
        [0, 71, 80, 12], [1, 71, 64, 216], [0, 64, 80, 24], [1, 64, 64, 432], [0, 76, 80, 48],
        [1, 76, 64, 216], [0, 71, 80, 24], [1, 71, 64, 108], [0, 74, 80, 12], [1, 74, 64, 108],
        [0, 76, 80, 12], [1, 76, 64, 216], [0, 76, 80, 24], [1, 76, 64, 216], [0, 71, 80, 24],
        [1, 71, 64, 108], [0, 74, 80, 12], [1, 74, 64, 108], [0, 76, 80, 12], [1, 76, 64, 216],
        [0, 76, 80, 24], [1, 76, 64, 432], [0, 76, 80, 48], [1, 76, 64, 216], [0, 71, 80, 24],
        [1, 71, 64, 108], [0, 74, 80, 12], [1, 74, 64, 108], [0, 76, 80, 12], [1, 76, 64, 216],
        [0, 76, 80, 24], [1, 76, 64, 216], [0, 71, 80, 24], [1, 71, 64, 108], [0, 74, 80, 12],
        [1, 74, 64, 108], [0, 76, 80, 12], [1, 76, 64, 216], [0, 76, 80, 24], [1, 76, 64, 432]]
mids2 = [[0, 37, 80, 0], [1, 37, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108], [0, 39, 80, 612],
         [1, 39, 64, 108], [0, 37, 80, 852], [1, 37, 64, 108], [0, 39, 80, 12], [1, 39, 64, 216],
         [0, 39, 80, 624], [1, 39, 64, 108], [0, 39, 80, 132], [1, 39, 64, 216], [0, 39, 80, 24],
         [1, 39, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108],
         [0, 37, 80, 252], [1, 37, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108], [0, 39, 80, 612],
         [1, 39, 64, 108], [0, 37, 80, 852], [1, 37, 64, 108], [0, 39, 80, 12], [1, 39, 64, 216],
         [0, 39, 80, 624], [1, 39, 64, 108], [0, 39, 80, 132], [1, 39, 64, 216], [0, 39, 80, 24],
         [1, 39, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108],
         [0, 37, 80, 732], [1, 37, 64, 108],[0, 37, 80, 0], [1, 37, 64, 108], [0, 39, 80, 12],
         [1, 39, 64, 108], [0, 39, 80, 612],[1, 39, 64, 108], [0, 37, 80, 852], [1, 37, 64, 108],
         [0, 39, 80, 12], [1, 39, 64, 216],[0, 39, 80, 624], [1, 39, 64, 108], [0, 39, 80, 132],
         [1, 39, 64, 216], [0, 39, 80, 24], [1, 39, 64, 108], [0, 39, 80, 12], [1, 39, 64, 108],
         [0, 39, 80, 12], [1, 39, 64, 108],[0, 37, 80, 252], [1, 37, 64, 108], [0, 39, 80, 12],
         [1, 39, 64, 108], [0, 39, 80, 612],[1, 39, 64, 108], [0, 37, 80, 852], [1, 37, 64, 108],
         [0, 39, 80, 12], [1, 39, 64, 216]]
# 播放MIDI
def play_music(music_file):
    clock = pg.time.Clock()
    pg.mixer.music.load(music_file)
    pg.mixer.music.play()
    while pg.mixer.music.get_busy():
        clock.tick(30)
# 主程序
fileName = "bcgm.mid"
mid = MidiFile()  # 创建midi文件
pg.mixer.init()
# 创建音轨
track1 = play_note(2,mids1,0)
track2 = play_note(44,mids2,4)
mid.tracks.append(track1)
mid.tracks.append(track2)
# 保存mid文件
mid.save(fileName)
play_music(fileName)


复制粘贴代码保存到Python文件中,运行Python文件,就能听到歌声啦~

目录
相关文章
|
3月前
|
存储 算法 调度
【复现】【遗传算法】考虑储能和可再生能源消纳责任制的售电公司购售电策略(Python代码实现)
【复现】【遗传算法】考虑储能和可再生能源消纳责任制的售电公司购售电策略(Python代码实现)
197 26
|
3月前
|
测试技术 开发者 Python
Python单元测试入门:3个核心断言方法,帮你快速定位代码bug
本文介绍Python单元测试基础,详解`unittest`框架中的三大核心断言方法:`assertEqual`验证值相等,`assertTrue`和`assertFalse`判断条件真假。通过实例演示其用法,帮助开发者自动化检测代码逻辑,提升测试效率与可靠性。
361 1
|
3月前
|
机器学习/深度学习 算法 调度
基于多动作深度强化学习的柔性车间调度研究(Python代码实现)
基于多动作深度强化学习的柔性车间调度研究(Python代码实现)
192 1
|
2月前
|
测试技术 Python
Python装饰器:为你的代码施展“魔法”
Python装饰器:为你的代码施展“魔法”
250 100
|
2月前
|
开发者 Python
Python列表推导式:一行代码的艺术与力量
Python列表推导式:一行代码的艺术与力量
407 95
|
3月前
|
Python
Python的简洁之道:5个让代码更优雅的技巧
Python的简洁之道:5个让代码更优雅的技巧
255 104
|
3月前
|
开发者 Python
Python神技:用列表推导式让你的代码更优雅
Python神技:用列表推导式让你的代码更优雅
453 99
|
2月前
|
缓存 Python
Python装饰器:为你的代码施展“魔法
Python装饰器:为你的代码施展“魔法
155 88
|
3月前
|
IDE 开发工具 开发者
Python类型注解:提升代码可读性与健壮性
Python类型注解:提升代码可读性与健壮性
281 102
|
2月前
|
监控 机器人 编译器
如何将python代码打包成exe文件---PyInstaller打包之神
PyInstaller可将Python程序打包为独立可执行文件,无需用户安装Python环境。它自动分析代码依赖,整合解释器、库及资源,支持一键生成exe,方便分发。使用pip安装后,通过简单命令即可完成打包,适合各类项目部署。

推荐镜像

更多