0

0

使用 PyAudio 播放声音并根据按键释放停止播放

碧海醫心

碧海醫心

发布时间:2025-08-21 19:38:27

|

463人浏览过

|

来源于php中文网

原创

 使用 PyAudio 播放声音并根据按键释放停止播放

本文介绍如何使用 PyAudio 库生成和播放声音,并根据 MIDI 输入的按键释放事件停止声音的播放。我们将分析一个现有的代码示例,并提供修改建议,使其能够响应按键释放事件,实现更灵活的声音控制。 ### 理解问题 原始代码存在的问题在于,它只能播放固定时长的声音,无法根据 MIDI 输入的按键释放事件停止声音。代码尝试通过 `np.arange` 删除持续时间,但没有解决问题。核心问题在于播放声音的循环结构和停止声音的逻辑。 ### 解决方案 核心思路是移除不必要的循环,并在按键释放时停止音频流。以下是修改后的代码片段和详细解释: **1. 移除不必要的循环** 原始代码中,`while play == True:` 循环只执行一次,然后立即停止。这个循环是不必要的,应该移除。 ```python # 原代码 while play == True: start_time = time.time() stream.write(vysledek) time.sleep(1) play = False

修改为:

stream.write(vysledek)

2. 监听按键释放事件

需要修改代码,监听MIDI输入的按键释放事件。MIDI 消息通常包含一个状态字节(例如 144 表示按键按下,128 表示按键释放)和一个数据字节(表示按键的编号)。

假设按键释放事件的状态字节是 128,我们可以添加以下代码来检测它:

if lepsi_klic in frekvence_seznam:
    # ... (生成声音的代码) ...

    stream.write(vysledek) # 播放声音

elif lepsi_klic == (128, message[0][1]): # 假设按键释放事件的状态字节是 128
    stream.stop_stream() # 停止声音
    print("停止播放")

elif lepsi_klic == (144, 81):
    exit()

3. 完整代码示例

以下是修改后的代码示例,包含上述修改:

LongShot
LongShot

LongShot 是一款 AI 写作助手,可帮助您生成针对搜索引擎优化的内容博客。

下载
import time
from rtmidi.midiutil import open_midiinput
import numpy as np
import pyaudio

p = pyaudio.PyAudio()

volume = 0.5  # range [0.0, 1.0]
fs = 44100  # sampling rate, Hz, must be integer
duration = 5.0  # in seconds, may be float
fA = 440.0  # sine frequency, Hz, may be float
fB = 493.88
fC = 523.25
fD = 587.33

frekvence = 440


frekvence_seznam = {
    (144, 32): fA,
    (144, 33): fB,
    (144, 34): fC,
    (144, 35): fD,


}

port = 0

midiin, port_name = open_midiinput(port)

stream = None  # 初始化 stream 变量

while True:
    msg = midiin.get_message()

    if msg:
        message = msg
        klic = message[0]
        lepsi_klic = tuple(klic[:2])
        print(message[0])

        if lepsi_klic in frekvence_seznam:
            print("开始播放")
            frekvence = frekvence_seznam[lepsi_klic]

            period = 2 * np.pi
            x = period * np.arange(fs * duration) * frekvence / fs

            sinus = np.sin(x)
            square = np.sign(sinus)
            triangle = 2/np.pi * np.arcsin(np.sin(x))
            saw = abs((x % period) - 1)
            curvy_triangle = (abs((x % period) - 1)) ** 2

            samples = (triangle).astype(np.float32)

            vysledek = volume * samples

            # 确保 stream 只创建一次
            if stream is None or not stream.is_active():
                stream = p.open(format=pyaudio.paFloat32,
                                channels=1,
                                rate=fs,
                                output=True)

            stream.write(vysledek) # 播放声音

        elif lepsi_klic[0] == 128: # 假设按键释放事件的状态字节是 128
            if stream is not None and stream.is_active():
                stream.stop_stream() # 停止声音
                print("停止播放")

        elif lepsi_klic == (144, 81):
            if stream is not None and stream.is_active():
                stream.stop_stream()
            stream.close()
            p.terminate()
            exit()

4. 代码解释

  • stream = None: 在 while 循环之前初始化 stream 变量。
  • if stream is None or not stream.is_active():: 检查 stream 是否已经创建并且处于活动状态。如果 stream 为 None 或者已经停止,则创建一个新的 stream。这确保了在按键按下时,如果 stream 不存在或已停止,会创建一个新的 stream。
  • elif lepsi_klic[0] == 128:: 监听状态字节为 128 的 MIDI 消息,这通常表示按键释放事件。
  • if stream is not None and stream.is_active():: 确保 stream 存在并且处于活动状态,然后再停止它。这避免了在 stream 未创建或已经停止时尝试停止 stream 导致的错误。
  • stream.stop_stream(): 停止音频流。
  • stream.close(): 在程序退出前关闭音频流。
  • p.terminate(): 在程序退出前终止 PyAudio 实例。

注意事项:

  • 确保安装了必要的库:pip install rtmidi numpy pyaudio
  • 根据实际 MIDI 设备发送的按键释放事件的状态字节修改代码中的 128。 可以通过打印 message[0] 的值来确定正确的状态字节。
  • 代码中的 duration 变量仍然存在,但它只影响 np.arange 创建的数组的大小。可以通过修改 duration 来调整声音的长度,或者完全移除它,并使用其他方法来控制声音的长度。

总结

通过移除不必要的循环并监听 MIDI 输入的按键释放事件,我们可以实现根据按键释放停止声音的播放。修改后的代码更加灵活,可以更好地控制声音的播放。 记住要根据你的 MIDI 设备和需求调整代码中的参数和逻辑。

					

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pip安装使用方法
pip安装使用方法

安装步骤:1、确保Python已经正确安装在您的计算机上;2、下载“get-pip.py”脚本;3、按下Win + R键,然后输入cmd并按下Enter键来打开命令行窗口;4、在命令行窗口中,使用cd命令切换到“get-pip.py”所在的目录;5、执行安装命令;6、验证安装结果即可。大家可以访问本专题下的文章,了解pip安装使用方法的更多内容。

339

2023.10.09

更新pip版本
更新pip版本

更新pip版本方法有使用pip自身更新、使用操作系统自带的包管理工具、使用python包管理工具、手动安装最新版本。想了解更多相关的内容,请阅读专题下面的文章。

415

2024.12.20

pip设置清华源
pip设置清华源

设置方法:1、打开终端或命令提示符窗口;2、运行“touch ~/.pip/pip.conf”命令创建一个名为pip的配置文件;3、打开pip.conf文件,然后添加“[global];index-url = https://pypi.tuna.tsinghua.edu.cn/simple”内容,这将把pip的镜像源设置为清华大学的镜像源;4、保存并关闭文件即可。

761

2024.12.23

python升级pip
python升级pip

本专题整合了python升级pip相关教程,阅读下面的文章了解更多详细内容。

349

2025.07.23

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

778

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

97

2023.09.25

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

18

2026.01.29

clawdbot龙虾机器人官网入口 clawdbot ai官方网站地址
clawdbot龙虾机器人官网入口 clawdbot ai官方网站地址

clawdbot龙虾机器人官网入口:https://clawd.bot/,clawdbot ai是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

12

2026.01.29

Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

8

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号