Vosk实战指南:打造你的离线语音识别系统

在智能语音时代,语音识别不再是大厂的专属。你是否也想打造一个无需联网、实时高效、支持中文的语音识别系统?本文将带你从零上手开源神器 Vosk ,实现本地语音转文字,即使在树莓派等设备上也能运行顺畅。

一、什么是Vosk?

Vosk 是一个轻量级的离线语音识别引擎,支持多种语言,拥有跨平台的优势,能在低资源设备上实现实时识别。最棒的是,它完全开源,适合个人开发者、教育科研、边缘AI等场景使用。

Vosk 的核心优势包括:

  • 离线运行,无需联网,保障隐私

  • 支持中文等多种语言

  • 可运行于 Windows、Linux、macOS、Android、树莓派等平台

  • 资源占用低,适配低功耗设备

  • Python、C++、Java、Node.js等多语言接口支持

二、Vosk 适合用来做什么?

你可以用 Vosk 构建各种本地语音识别应用,例如:

  • 桌面语音笔记工具

  • 车载语音助手

  • 儿童教育类语音交互玩具

  • 语音触发报警系统

  • 面向听障人群的语音字幕提示器

如果你正在做AI自媒体内容、边缘AI设备开发、或者是语音UI设计,Vosk是一个值得深入了解的方案。

三、安装与环境准备

我们以 Python 环境为例进行快速上手,我测试的系统是Windows系统。

安装依赖

bash 复制代码
pip install vosk sounddevice

# sounddevice 用于从麦克风获取音频流。

下载中文模型

前往 https://alphacephei.com/vosk/models 下载中文模型,推荐从 vosk-model-small-cn-0.22 开始入手,解压到当前目录,我这里使用默认命名vosk-model-small-cn-0.22

四、实战演示:实时中文语音转文字

下面是一个实时麦克风识别的完整Python示例。

python 复制代码
import sounddevice as sd
import queue
import json
from vosk import Model, KaldiRecognizer

q = queue.Queue()

def callback(indata, frames, time, status):
    q.put(bytes(indata))

# 加载模型
model = Model("vosk-model-small-cn-0.22") # 使用自己下载好的模型路径
recognizer = KaldiRecognizer(model, 16000)

# 打开麦克风输入
with sd.RawInputStream(samplerate=16000, blocksize=8000, dtype='int16',
                       channels=1, callback=callback):
    print("请开始说话(Ctrl+C 结束)")
    while True:
        data = q.get()
        if recognizer.AcceptWaveform(data):
            result = json.loads(recognizer.Result())
            print("识别结果:", result.get("text"))

只需几行代码,就能实现语音实时转文字,而且完全离线无需联网

五、其他相关用法

  • 识别结果延迟调整

尝试减小 blocksize 或优化麦克风质量。

  • 指定关键词识别

Vosk 支持关键词语法,例如限制识别词汇为 ["打开灯", "关闭灯"]

python 复制代码
KaldiRecognizer(model, 16000, '["打开灯", "关闭灯"]')
  • 音频文件识别,这里只能是wav文件格式

可以用 wave 模块读取 .wav 文件输入到 recognizer 中。

六、总结

Vosk 是一个实用、可靠、开源的语音识别工具。如果你:

  • 想做语音类自媒体项目

  • 想开发边缘设备语音UI

  • 想离线转写会议记录

  • 想构建自己的语音助手

那么 Vosk 完全值得尝试。无论你是AI开发者,还是正在寻找语音识别解决方案的产品人,Vosk 都可能成为你项目中的关键一环。

七、延伸阅读推荐

附语音与音频文件语音识别工程代码(完整详细版):

python 复制代码
import os
import sys
import json
import queue
import re
import wave
import argparse
import subprocess
import tempfile
import sounddevice as sd
from typing import Optional, List
from vosk import Model, KaldiRecognizer

# 常量定义
AUDIO_SAMPLE_RATE = 16000
AUDIO_CHANNELS = 1
AUDIO_SAMPLE_WIDTH = 2  # 16位
FRAMES_PER_CHUNK = 4000

class AudioProcessor:
    @staticmethod
    def clean_spaces(text: str) -> str:
        """清理文本中的多余空格"""
        text = re.sub(r'([\u4e00-\u9fa5])\s+([a-zA-Z0-9])', r'\1\2', text)
        text = re.sub(r'([a-zA-Z0-9])\s+([\u4e00-\u9fa5])', r'\1\2', text)
        text = re.sub(r'\s+', ' ', text)
        return text.strip()

    @staticmethod
    def convert_audio(input_file: str, output_file: str) -> bool:
        """转换音频为Vosk兼容格式"""
        print(f"正在转换音频: {input_file} -> {output_file}")
        try:
            subprocess.run([
                "ffmpeg", "-y", "-i", input_file,
                "-ar", str(AUDIO_SAMPLE_RATE),
                "-ac", str(AUDIO_CHANNELS),
                "-sample_fmt", "s16",
                output_file
            ], check=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
            return True
        except (subprocess.CalledProcessError, FileNotFoundError) as e:
            print(f"音频转换失败: {e}")
            return False

class VoskRecognizer:
    def __init__(self, model_path: str):
        if not os.path.exists(model_path):
            raise FileNotFoundError(f"模型目录不存在: {model_path}")
        self.model = Model(model_path)

    def recognize_file(self, filename: str) -> Optional[str]:
        """识别音频文件内容"""
        try:
            # 尝试打开音频文件
            try:
                wf = wave.open(filename, "rb")
            except wave.Error:
                # 创建resources/temp目录如果不存在
                temp_dir = os.path.join(os.path.dirname(__file__), "resources", "temp")
                os.makedirs(temp_dir, exist_ok=True)
            
                # 生成临时文件路径
                temp_path = os.path.join(temp_dir, "converted_temp.wav")

                
                if not AudioProcessor.convert_audio(filename, temp_path):
                    return None
                
                filename = temp_path
                wf = wave.open(filename, "rb")
                print("音频转换成功,开始识别...")

            # 检查音频格式
            if not self._check_audio_format(wf):
                print("音频格式不符合要求")
                return None

            # 开始识别
            rec = KaldiRecognizer(self.model, wf.getframerate())
            results = []

            while True:
                data = wf.readframes(FRAMES_PER_CHUNK)
                if not data:
                    break
                if rec.AcceptWaveform(data):
                    res = json.loads(rec.Result())
                    results.append(res.get("text", ""))

            final_text = " ".join(results)
            return AudioProcessor.clean_spaces(final_text)

        except Exception as e:
            print(f"识别过程中出错: {e}")
            return None
        finally:
            if 'wf' in locals():
                wf.close()
            if 'temp_path' in locals() and os.path.exists(temp_path):
                os.unlink(temp_path)

    def recognize_mic(self) -> None:
        """从麦克风实时识别"""
        q = queue.Queue()

        def callback(indata, frames, time, status):
            q.put(bytes(indata))

        rec = KaldiRecognizer(self.model, AUDIO_SAMPLE_RATE)
        print("请开始说话,按 Ctrl+C 退出:")

        try:
            with sd.RawInputStream(
                samplerate=AUDIO_SAMPLE_RATE,
                blocksize=8000,
                dtype='int16',
                channels=AUDIO_CHANNELS,
                callback=callback
            ):
                while True:
                    data = q.get()
                    if rec.AcceptWaveform(data):
                        res = json.loads(rec.Result())
                        text = res.get("text", "")
                        cleaned = AudioProcessor.clean_spaces(text)
                        print(f"识别结果: {cleaned}")
        except KeyboardInterrupt:
            print("\n录音识别已终止。")
        except Exception as e:
            print(f"麦克风识别出错: {e}")

    def _check_audio_format(self, wf: wave.Wave_read) -> bool:
        """检查音频格式是否符合要求"""
        return (
            wf.getnchannels() == AUDIO_CHANNELS and
            wf.getsampwidth() == AUDIO_SAMPLE_WIDTH and
            wf.getframerate() == AUDIO_SAMPLE_RATE
        )

def main():
    parser = argparse.ArgumentParser(description="Vosk 语音识别工具", formatter_class=argparse.ArgumentDefaultsHelpFormatter)
    parser.add_argument("--mode", choices=["mic", "file"], required=True, help="选择识别模式")
    parser.add_argument("--file", help="音频文件路径(file模式需要)")
    parser.add_argument("--model", default="models/vosk-model-small-cn-0.22", help="模型目录路径")
    
    args = parser.parse_args()

    try:
        recognizer = VoskRecognizer(args.model)
        
        if args.mode == "mic":
            recognizer.recognize_mic()
        elif args.mode == "file":
            if not args.file:
                print("错误: file模式需要指定--file参数")
                return
                
            if not os.path.isfile(args.file):
                print(f"错误: 文件不存在 - {args.file}")
                return
                
            result = recognizer.recognize_file(args.file)
            if result:
                print(f"\n最终识别结果:\n{result}")
    except Exception as e:
        print(f"程序运行出错: {e}")
        sys.exit(1)

if __name__ == "__main__":
    main()
相关推荐
梦梦代码精5 小时前
为什么这个开源的AI平台会火?有点东西。。。
人工智能·算法·机器学习·docker·开源
大模型真好玩5 小时前
智能体从入门到精通:6个必学GitHub开源项目
人工智能·agent·deepseek
源图客6 小时前
Aitoearn:OPC(一人公司)的AI内容智能体
人工智能·dreamweaver
逸模6 小时前
AI+BIM 重构连锁公装新范式 逸模打造数字化营建核心底座
大数据·人工智能·笔记·其他·信息可视化·重构
phltxy6 小时前
MCP 从协议到 Spring AI 实战
人工智能·spring·oracle
Sirius Wu6 小时前
Agentic端到端&分离式RL技术建设
人工智能·深度学习·机器学习·caffe
AI导出鸭PC端7 小时前
智谱清言怎么生成word文档?AI导出鸭终结乱码烦恼
人工智能·ai·c#·word·豆包·ai导出鸭
格桑阿sir7 小时前
17-大模型智能体开发工程师:深入学习Agent记忆系统
人工智能·记忆存储·记忆系统·agent记忆·嵌入式数据库·agent进化·记忆检索
数据仓库搬砖人7 小时前
LangGraph 原理深度解析:为什么它是目前最适合构建 Agent 的框架
人工智能
孟陬7 小时前
国外技术周刊 #139:LLM 正在杀死程序员的「懒惰美德」
前端·人工智能·后端