开发完成语音播报产品,由于客户所使用的播放产品种类繁多,在使用HDMI接口播放音频时,由于采用的声卡不同,个别机器会出现播报声音过小,或者不播报的情况。所以采用将语音文件合并播放的方式,来解决此问题。

   /// <summary>
        /// 合并语音流
        /// </summary>
        /// <param name="pPlayStreamsList">语音流集合</param>
        public void MergeAndPlayWavFile(ObservableCollection<Stream> pPlayStreamsList)
        {
            try
            {
                if (pPlayStreamsList.Count > 0)
                {
                    int dataFileBeginIndex = 0;
                    int formatSize;
                    List<byte[]> bInfoList = new List<byte[]>();
                    List<int> fileSizeList = new List<int>();
                    List<int> dataSizeList = new List<int>();
                    foreach (Stream fs in pPlayStreamsList)
                    {
                        fs.Position = 0;
                        byte[] bInfo = new byte[46];
                        fs.Read(bInfo, 0, 46);
                        formatSize = BitConverter.ToInt32(bInfo, 16);
                        //Wave = RIFF_WAVE_Chunk + Format_Chunk + Data_Chunk(RIFF)/Fact_Chunk(FACT)
                        //RIFF_WAVE_Chunk(文件头,12个字节)+
                        //Format_Chunk(声音内容定义,根据第16到20个字节得到剩下区块大小,16或者18个字节)+
                        //Data_Chunk(数据区)
                        dataFileBeginIndex = formatSize + 20 + 4;
                        bInfoList.Add(bInfo);
                        fileSizeList.Add(System.BitConverter.ToInt32(bInfo, 4));
                        dataSizeList.Add(System.BitConverter.ToInt32(bInfo, dataFileBeginIndex));
                    }

                    //计算所有WAV文件大小
                    int fileSize = 0;
                    foreach (int fSize in fileSizeList)
                    {
                        fileSize += fSize;
                    }
                    byte[] totalFileSize = System.BitConverter.GetBytes(fileSize);

                    //计算所有WAV数据大小
                    int dataSize = 0;
                    foreach (int dSize in dataSizeList)
                    {
                        dataSize += dSize;
                    }
                    byte[] totalDataSize = System.BitConverter.GetBytes(dataSize);
                    using (Stream toFile = new MemoryStream())
                    {
                        //Stream toFile = new MemoryStream();
                        BinaryWriter bWriter = new BinaryWriter(toFile);

                        //获取最后一个wav文件的详细内容
                        int lastWaveFile = pPlayStreamsList.Count - 1;

                        for (int i = 4, j = 0; i < 8; i++, j++)
                        {
                            bInfoList[lastWaveFile][i] = totalFileSize[j];
                        }

                        for (int i = dataFileBeginIndex, j = 0; i < dataFileBeginIndex + 4; i++, j++)
                        {
                            bInfoList[lastWaveFile][i] = totalDataSize[j];
                        }
                        bWriter.Write(bInfoList[lastWaveFile]);
                        int k = 0;
                        foreach (Stream fs in pPlayStreamsList)
                        {
                            int readLength;
                            fs.Position = dataFileBeginIndex + 4;
                            if (k < dataSizeList.Count)
                            {
                                readLength = dataSizeList[k] - 40;
                                byte[] dushu = new byte[readLength];
                                fs.Read(dushu, 0, readLength);
                                bWriter.Write(dushu, 0, readLength);
                                k++;
                            }
                        }
                        SoundPlayer player = new SoundPlayer();
                        if (toFile != null)
                        {
                            player.Stream = null;
                            player.Stream = toFile;
                            toFile.Position = 0;
                            player.LoadAsync();
                            for (int i = 0; i < playTimes; i++)
                            { player.PlaySync(); }

                        }

                        bWriter.Flush();
                        bWriter.Close();

                        toFile.Flush();
                        toFile.Close();
                        player.Dispose();
                        bWriter.Dispose();
                        toFile.Dispose();
                    }
                }
            }
            catch (Exception e)
            {
                LogInfo.saveLog("合并语音流报错,方法MergeAndPlayWavFile():" + e.Message);
            }

   }

C# wav语音文件合并的更多相关文章

  1. [Audio processing] wav音频文件合并

    合并多个文件,需要包含1.文件读取和写入功能,2.数组合并 package com.audioprocessingbox.myfunc; import java.io.File; import jav ...

  2. 多个wav音频文件合并(连接)成一个文件

    场景:一段声音从浏览器麦克风缓冲上一段一段发给服务器,按照时间戳生成很多文件. 目的:把他们按时间顺序连到一个时间轴上. 命令如下: ffmpeg -f concat -i list.txt out. ...

  3. Python网页正文转换语音文件的操作方法

    天气真的是越来越冷啦,有时候我们想翻看网页新闻,但是又冷的不想把手拿出来,移动鼠标翻看.这时候,是不是特别想电脑像讲故事一样,给我们念出来呢?人生苦短,我有python啊,试试用 Python 来朗读 ...

  4. WAV格式文件无损合并&帧头数据体解析(python)(原创)

    一,百度百科 WAV为微软公司(Microsoft)开发的一种声音文件格式,它符合RIFF(Resource Interchange File Format)文件规范,用于保存Windows平台的音频 ...

  5. C# 使用NAudio合并mp3、wav音频文件

    1.什么是wav格式    WAV为微软公司(Microsoft)开发的一种声音文件格式,它符合RIFF(Resource Interchange File Format)文件规范,用于保存Windo ...

  6. asp.net使用SpeechSynthesizer类生成语音文件部署到iis遇到的几个坑

    首先需要引入命名空间System.Speech.Synthesis,代码如下: using (var speechSyn = new SpeechSynthesizer()) { speechSyn. ...

  7. 解析WAV音频文件----》生成WAV音频文件头

    前言:请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i WAV音频文件介绍: WAV文件是在PC机平台上很常见的.最经典的多媒体音频文件,最早于1991年8月出现在Windows3.1操作系统 ...

  8. CDN的combo技术能把多个资源文件合并引用,减少请求次数

    CDN的combo技术能把多个资源文件合并引用,减少请求次数.比如淘宝的写法: <link rel="stylesheet" href="//g.alicdn.co ...

  9. linux 两个文件合并

    可以使用cat命令,有两种实现的方式,一种将两个文件合并的到一个新的文件,另一种将一个文件追加到另一个文件的末尾. 方法一:使用cat命令从文件中读入两个文件,然后将重定向到一个新的文件.这种方法可以 ...

随机推荐

  1. BZOJ 3262: 陌上花开 [CDQ分治 三维偏序]

    Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当 ...

  2. 联合查询到gridview

    using com.DAL.Base; using DAL.ruanmou; using System; using System.Collections.Generic; using System. ...

  3. 使用Python的requests库进行接口测试——session对象的妙用

    from:http://blog.csdn.net/liuchunming033/article/details/48131051 在进行接口测试的时候,我们会调用多个接口发出多个请求,在这些请求中有 ...

  4. python数据分析工具包(2)——Numpy(二)

    上一篇文章简单地介绍了numpy的一些基本数据类型,以及生成数组和矩阵的操作.下面我们来看一下矩阵的基本运算.在线性代数中,常见的矩阵运算包括,计算行列式.求逆矩阵.矩阵的秩等.下面我们来一一实现. ...

  5. 嵌入式linux系统的构建

    前期工作:a.配置好tftp服务器:在嵌入式的童年中有介绍 b.开发板可以pc,linux 三者可以互相ping通 c.配置好nfs服务器:同样在嵌入式的童年中有介绍 一.嵌入式linux内核的制作( ...

  6. 基于Java的WebSocket推送

    WebSocket的主动推送 关于消息推送,现在的解决方案如轮询.长连接或者短连接,当然还有其他的一些技术框架,有的是客户端直接去服务端拿数据. 其实推送推送主要讲的是一个推的概念,WebSocket ...

  7. System.in实现数据的键盘输入

    System.in The "standard" input stream. This stream is already open and ready to supply inp ...

  8. SSE图像算法优化系列十七:多个图像处理中常用函数的SSE实现。

    在做图像处理的SSE优化时,也会经常遇到一些小的过程.数值优化等代码,本文分享一些个人收藏或实现的代码片段给大家. 一.快速求对数运算 对数运算在图像处理中也是个经常会遇到的过程,特备是在一些数据压缩 ...

  9. Jenkins gitlab vue,angular,react 自动化构建【原】

    大致思路,(本篇主要讲vue ,当然了 angular react 也是一样配置) ,转发请注明原链接,谢谢 :) 1. 服务器上面配置jenkins (安装配置,不介绍) 2.新建item 自由风格 ...

  10. hibernate之实体@onetomany和@manytoone双向注解(转)

    下面是User类: @onetomany @Entity @Table(name="user") public class User implements Serializable ...