开发完成语音播报产品,由于客户所使用的播放产品种类繁多,在使用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. 整理一点与排列组合有关的问题[组合数 Stirling数 Catalan数]

    都是数学题 思维最重要,什么什么数都没用,DP直接乱搞(雾.. 参考LH课件,以及资料:http://daybreakcx.is-programmer.com/posts/17315.html 做到有 ...

  2. c语言环境初始化&c语言和汇编混合编程

    bootloader通常会分为两个阶段:第一阶段采用汇编语言来编写,主要是一些核心的初始化工作(内存,时钟的初始化),第二阶段使用C语言来编写,主要是它会完成一些板载硬件的初始化(串口,网口)然后其启 ...

  3. Spring基础篇——通过Java注解和XML配置装配bean

    自动化装配的确有很大的便利性,但是却并不能适用在所有的应用场景,比如需要装配的组件类不是由自己的应用程序维护,而是引用了第三方的类库,这个时候自动装配便无法实现,Spring对此也提供了相应的解决方案 ...

  4. 【linux之shell脚本】

    一.简介 机器语言汇编语言高级语言 面向过程 C Shell Perl 面向对象 java python c++ 强语言:先编译再执行 java c++ 弱语言:边编译边执行 shell python ...

  5. mysql 修改默认字符集为utf8

    MySQL 5.5, all you need is: [mysqld] character_set_client=utf8 character_set_server=utf8 collation_s ...

  6. linux开放80 端口

    1.使用su登录管理员用户 2.编辑防火墙配置文件 vim /etc/sysconfig/iptables 3.在里面加入后保存 #open port 80 -A INPUT -p TCP -m st ...

  7. 在web工程中设置首页的页面

    有些时候删除了系统自带的index.jsp删除后会出现如下图错误 解决办法,新创建一个以你自己命名的jsp文件,然后在对该web工程的WEB-INF 目录下的web.xml进行添加加上下面的注释所带的 ...

  8. java打包项目将配置文件放在包外面(后续还会有补充)

    项目中也经常单独将一部分功能独立做Java Project,然后打成jar包供其他项目调用.如果jar包中需要读取配置文件信息,则很少把该配置打进jar包,因为它不方便修改,更多都是采用jar包读取外 ...

  9. uva 1583

    枚举1~100000把所有数的最小generators记录下来,直接查表即可. AC代码: #include<cstdio> #include<cstring> #includ ...

  10. LOJ6000 - 「网络流 24 题」搭配飞行员

    原题链接 题意简述 求二分图的最大匹配. 题解 这里写的是匈牙利算法. 表示节点的当前匹配. 为真表示在这一轮匹配中,无法给节点一个新的匹配.所以如果为真就不用再dfs它了,直接continue就好. ...