开发完成语音播报产品,由于客户所使用的播放产品种类繁多,在使用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. 自己动手实践 spring retry 重试框架

    前序 马上过年了,预祝大家,新年快乐,少写bug 什么是spring retry? spring retry是从spring batch独立出来的一个能功能,主要实现了重试和熔断. 什么时候用? 远程 ...

  2. 关于.NET C#上传大文件的解决办法

    1.最近在解决问题的时候遇到如何将视频以及语音和图片上传到阿里云的服务器中.但是遇到一些大文件就导致无法进行上传. 2.在将图片进行上传到阿里云的时候先将文件转化为二进制文件,然后通过文件流 的形式进 ...

  3. 网络编程基础+UDP的实现

    网络地址分类(32位网络地址) A类地址:第一个字节为网络地址,其他3个字节主机地址,第一字节的最高位固定为0 从1.0.0.1------126.255.255.255 B类地址:第一字节和第二字节 ...

  4. 关于一些php规范

    <?php /** * 符合psr-1,2的编程实例 * * @author GreenForestQuan */ namespace Standard; // 顶部命名空间 // 空一行 us ...

  5. [Python Study Notes]CS架构远程访问获取信息--SERVER端v2.0

    更新内容: 1.增加内存信息获取 2.增加电池信息获取 3.增加磁盘信息获取 4.重新布局窗体 5.增加窗体名称 6.增加连接成功之前,不可按压 ''''''''''''''''''''''''''' ...

  6. MarkDown思考

    前言 使用MarkDown有一段时间了,感觉的确很方便.大大提高了工作效率,并带来了良好的工作体验.但是,一直以来我都有一个困惑,就是MarkDown的插件和编辑器纷繁无比,却通常各自有一套自己的实现 ...

  7. IOS中DES与MD5加密方案

      0 2 项目中用的的加密算法,因为要和安卓版的适配,中间遇到许多麻烦. MD5算法和DES算法是常见的两种加密算法. MD5:MD5是一种不可逆的加密算法,按我的理解,所谓不可逆,就是不能解密,那 ...

  8. docker 实践(一)

    docker 简介 容器虚拟化,比传统的虚拟化轻量 2013年出现,发展非常迅猛 Redhat在6.5版本开始支持docker 使用go语言开发,基于apache2.0协议 开源软件,项目代码在git ...

  9. Mysql数据库查询不区分大小写解决方案

  10. zip-gzip-bzip2_压缩文件

    问:为什么要压缩文件? 答:方便传输,因为压缩的文件容量会比较小        存储所使用的空间也会比较小 ---> 备份   Windows里的压缩软件:WinRAR.Zip.好压.2345 ...