Error in user YAML: (<unknown>): could not find expected ':' while scanning a simple key at line 3 column 1
---
- oeasy Python 0102
- 这是 oeasy 系统化 Python 教程,从基础一步步讲,扎实、完整、不跳步。愿意花时间学,就能真正学会。
本教程同步发布在:
个人网站: `https://oeasy.org`
蓝桥云课: `https://www.lanqiao.cn/courses/3584`
GitHub: `https://github.com/overmind1980/oeasy-python-tutorial`
Gitee: `https://gitee.com/overmind1980/oeasypython`
---- 配套视频
- 上次 研究了
回文诗- 对称 的 文字
- 可以用 列表反转 的 方式
- 进行判断
- 音符也可以 实现 类似于 回文的效果
- 卡农螃蟹曲式
- 能做段 简单的
回文音乐吗?🤔 - 怎么 才能 唱出 歌词 呢?
- 可以下载x-studio
- 让 虚拟歌姬 唱歌 应该分两步
- 在软件中 导入 开嗓旋律 的 mid
- 然后 右键音符 编辑全部歌词
- 去 生成mid
- 曲式的对称 很像 回文的对称
- 配合回文
- 能用代码 生成这个mid吗?
- 首先新建文件
- m.py
from mido import Message, MidiFile, MidiTrack
# 创建一个新的MIDI文件和音轨
mid = MidiFile()
track = MidiTrack()
mid.tracks.append(track)
# 定义C4对应的MIDI音符编号
c4_note = 60
# 定义每个音符的持续时间(这里设为0.5秒,可按需调整)
note_duration = 0.5
# 定义BPM(每分钟节拍数),控制节奏速度
bpm = 120
# 计算每拍对应的时间(秒)
tick_per_beat = mid.ticks_per_beat
time_per_beat = 60 / bpm
time_per_tick = time_per_beat / tick_per_beat
# 定义“哆来咪发嗦发咪来哆”对应的音符偏移量
# 基于C4(60号音符),在C大调中,各音符偏移量对应关系
note_offsets = [0, 2, 4, 5, 7, 5, 4, 2, 0]
# 遍历音符偏移量来添加音符事件到音轨
for offset in note_offsets:
note_number = c4_note + offset
# 按下音符事件
track.append(Message('note_on', note=note_number, velocity=64, time=0))
# 计算音符持续时间对应的tick数
tick_duration = int(note_duration / time_per_tick)
# 松开音符事件
track.append(Message('note_off', note=note_number, velocity=64, time=tick_duration))
# 保存MIDI文件
mid.save('warmup.mid')- 注意哼鸣练习中的 列表
- 依然是 回文结构的
- 选择 文件-打开
- 打开 刚下好的mid文件
- 目前是 4/4 拍
- 以 4分音符 为 1拍
- 每小节 有 4拍
- 最后一个小节
- 只有一个 四分音符
- 应该是 全音符
- 时长4拍
- 独占一个小节
from mido import Message, MidiFile, MidiTrack
# 创建一个新的MIDI文件和音轨
mid = MidiFile()
track = MidiTrack()
mid.tracks.append(track)
# 定义C4对应的MIDI音符编号
c4_note = 60
# 定义除最后一个音符外其他音符的持续时间(这里设为0.5秒,可按需调整)
note_duration = 0.5
# 定义BPM(每分钟节拍数),控制节奏速度
bpm = 120
# 计算每拍对应的时间(秒)
tick_per_beat = mid.ticks_per_beat
time_per_beat = 60 / bpm
time_per_tick = time_per_beat / tick_per_beat
# 定义“哆来咪发嗦发咪来哆”对应的音符偏移量
# 基于C4(60号音符),在C大调中,各音符偏移量对应关系
note_offsets = [0, 2, 4, 5, 7, 5, 4, 2, 0]
# 遍历音符偏移量来添加音符事件到音轨
for i, offset in enumerate(note_offsets[:-1]):
note_number = c4_note + offset
# 按下音符事件
track.append(Message('note_on', note=note_number, velocity=64, time=0))
# 计算音符持续时间对应的tick数
tick_duration = int(note_duration / time_per_tick)
# 松开音符事件
track.append(Message('note_off', note=note_number, velocity=64, time=tick_duration))
# 处理最后一个音符(全音符,四拍)
last_offset = note_offsets[-1]
last_note_number = c4_note + last_offset
# 按下最后一个音符事件
track.append(Message('note_on', note=last_note_number, velocity=64, time=0))
# 全音符时长对应的tick数(四拍)
tick_duration_for_last = int(4 * time_per_beat / time_per_tick)
# 松开最后一个音符事件
track.append(Message('note_off', note=last_note_number, velocity=64, time=tick_duration_for_last))
# 保存MIDI文件
mid.save('warmup2.mid')- 演唱一下
- 可以让 开嗓midi越开越高吗?
- 先复制一个开嗓 片段
- 然后将 片段 上移 一个半音
- 得到 C#大调 的 哼鸣练习
- 这样 第二段哼鸣 就升高了半音
- 1段 变 2段
- 2段 变 4段
- 4段 变 8段
- 8段 变 16段
- 删除 其中3段
- 还剩 13段
- 从c开始
- 到c结束
- 开嗓
- 想用代码 实现这个mid
from mido import Message, MidiFile, MidiTrack
# 基础参数
base_note = 60 # C4的MIDI音符编号(60对应C4)
note_offsets = [0, 2, 4, 5, 7, 5, 4, 2, 0] # 123454321的音程偏移(C大调)
bpm = 120
ticks_per_beat = 480 # 可自定义MIDI精度
time_per_beat = 60 / bpm
time_per_tick = time_per_beat / ticks_per_beat
note_duration = 0.5 # 前8个音符时长(秒),最后一个音为4拍
# 生成12个半音调的MIDI文件
for transpose in range(13): # 0(原调)到12(升12半音)
mid = MidiFile(ticks_per_beat=ticks_per_beat)
track = MidiTrack()
mid.tracks.append(track)
# 添加调号(仅为示意,实际MIDI调号需用SysEx信息,此处通过音符移位实现)
current_note = base_note + transpose # 基础音升高transpose个半音
for i, offset in enumerate(note_offsets):
note_number = current_note + offset
# 音符按下
track.append(Message('note_on', note=note_number, velocity=64, time=0))
# 计算时长(最后一个音为4拍)
if i == len(note_offsets) - 1:
tick_duration = int(4 * time_per_beat / time_per_tick)
else:
tick_duration = int(note_duration / time_per_tick)
# 音符松开
track.append(Message('note_off', note=note_number, velocity=64, time=tick_duration))
# 保存文件(调名对应:C, C#, D, ..., B)
sharps = ['C', 'C#', 'D', 'D#', 'E', 'F', 'F#', 'G', 'G#', 'A', 'A#', 'B', 'C'][transpose]
mid.save(f'scale_{sharps}_major.mid')
print(f"已生成 {sharps} 大调音阶 MIDI 文件")
- 生成了好多midi文件
- 大调 是啥意思?
- 这次我们 制作了回文音符序列
- 123454321
- 要整合成一个
- 序列不断升级
- 从c开始
- 到c结束
- 序列不断升级
- 可以整合吗?🤔
- 下次再说 👋
- 配套视频
- 本文来自 oeasy Python 系统教程。
- 想完整、扎实学 Python,
- 搜索 oeasy 即可。





















