C# 基于NAudio实现对Wav音频文件剪切(限PCM格式)
作者:QianTLL 发布时间:2022-08-31 09:55:18
标签:C#,NAudio,音频文件剪切
前言
C#基于NAudio工具对Wav音频文件进行剪切,将一个音频文件剪切成多个音频文件
注:调用方法前需要导入NAudio.dll或者在NuGet程序管理器搜索NAudio并安装
本文是按时间剪切
实现代码
using NAudio.Wave;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace XXX.util
{
public static class WavFileUtils
{
/// <summary>
/// 基于NAudio工具对Wav音频文件剪切(限PCM格式)
/// </summary>
/// <param name="inPath">目标文件</param>
/// <param name="outPath">输出文件</param>
/// <param name="cutFromStart">开始时间</param>
/// <param name="cutFromEnd">结束时间</param>
public static void TrimWavFile(string inPath, string outPath, TimeSpan cutFromStart, TimeSpan cutFromEnd)
{
using (WaveFileReader reader = new WaveFileReader(inPath))
{
int fileLength = (int)reader.Length;using (WaveFileWriter writer = new WaveFileWriter(outPath, reader.WaveFormat))
{
float bytesPerMillisecond = reader.WaveFormat.AverageBytesPerSecond / 1000f;
int startPos = (int)Math.Round(cutFromStart.TotalMilliseconds * bytesPerMillisecond);
startPos = startPos - startPos % reader.WaveFormat.BlockAlign;
int endPos = (int)Math.Round(cutFromEnd.TotalMilliseconds * bytesPerMillisecond);
endPos = endPos - endPos % reader.WaveFormat.BlockAlign;
//判断结束位置是否越界
endPos = endPos > fileLength ? fileLength : endPos;
TrimWavFile(reader, writer, startPos, endPos);
}
}
}
/// <summary>
/// 重新合并wav文件
/// </summary>
/// <param name="reader">读取流</param>
/// <param name="writer">写入流</param>
/// <param name="startPos">开始流</param>
/// <param name="endPos">结束流</param>
private static void TrimWavFile(WaveFileReader reader, WaveFileWriter writer, int startPos, int endPos)
{
reader.Position = startPos;
byte[] buffer = new byte[1024];
while (reader.Position < endPos)
{
int bytesRequired = (int)(endPos - reader.Position);
if (bytesRequired > 0)
{
int bytesToRead = Math.Min(bytesRequired, buffer.Length);
int bytesRead = reader.Read(buffer, 0, bytesToRead);
if (bytesRead > 0)
{
writer.Write(buffer, 0, bytesRead);
}
}
}
}
}
}
调用:
string filePath = "D:\\wav\\test.wav";//需要切割的文件路径
int cutTimeSpan = 20;//切割的时间片段时间(秒)
FileInfo fi = new FileInfo(filePath);
//获取录音文件时长(秒)
int fileTime = (int)Util.Cover(Util.GetVoiceTime(filePath)) / 1000;
//计算文件需要切割多少等份
decimal fileNum = Math.Ceiling((decimal)fileTime / cutTimeSpan);
int i = 0;
while (i < fileNum)
{
string nowTime = Util.GetTimeStamp();//当前时间戳
//切割后保存的文件绝对地址
var outputPath = System.IO.Path.Combine(fi.Directory.FullName, string.Format("{0}_{1}{2}", fi.Name.Replace(fi.Extension, ""), nowTime, fi.Extension));
//切割的开始时间
TimeSpan cutFromStart = TimeSpan.FromSeconds(i * cutTimeSpan);
//切割的结束时间
TimeSpan cutFromEnd = cutFromStart + TimeSpan.FromSeconds(cutTimeSpan);
//音频切割
WavFileUtils.TrimWavFile(recordFile.FilePath, outputPath, cutFromStart, cutFromEnd);
i++;
}
Util 类:
using Shell32;
using System;
using System.Diagnostics;
using System.IO;
using System.Net;
using System.Net.Sockets;
using System.Text.RegularExpressions;
using System.Threading;
using System.Windows.Forms;
namespace XXX.util
{
class Util
{
/// <summary>
/// 获取时间戳
/// </summary>
/// <returns></returns>
public static string GetTimeStamp()
{
TimeSpan ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0);
return Convert.ToInt64(ts.TotalMilliseconds).ToString();
}
/// <summary>
/// 返回音频时长
/// </summary>
/// <param name="SongPath">音频文件路径</param>
/// <returns></returns>
public static string GetVoiceTime(string SongPath)
{
string dirName = Path.GetDirectoryName(SongPath);
string SongName = Path.GetFileName(SongPath);
ShellClass sh = new ShellClass();
Folder dir = sh.NameSpace(dirName);
FolderItem item = dir.ParseName(SongName);
string SongTime = Regex.Match(dir.GetDetailsOf(item, -1), "\\d:\\d{2}:\\d{2}").Value;//返回音频时长
return SongTime;
}
/// <summary>
/// 时间格式转毫秒值
/// </summary>
/// <param name="time">时间字符串</param>
/// <returns></returns>
public static long Cover(string time)
{
string[] a = time.Split(':');
if (long.Parse(a[0]) == 0 && long.Parse(a[1]) == 0)
{
return long.Parse(a[2]) * 1000;
}
else if (long.Parse(a[0]) == 0 && long.Parse(a[1]) != 0)
{
return (long.Parse(a[1]) * 60 + long.Parse(a[2])) * 1000;
}
else if (long.Parse(a[0]) != 0 && long.Parse(a[1]) == 0)
{
return ((long.Parse(a[0]) * 60 * 60) + long.Parse(a[2])) * 1000;
}
else if (long.Parse(a[0]) != 0 && long.Parse(a[1]) != 0)
{
return (((long.Parse(a[0]) * 60) + long.Parse(a[1])) * 60) * 1000;
}
return 0;
}
}
}
效果图
来源:https://www.cnblogs.com/qiantao/archive/2021/11/29/15618359.html


猜你喜欢
- JPA的加锁机制有两种,乐观锁和悲观锁。乐观锁:乐观锁的特点在于认为数据冲突或者更新丢失等情况是很少发生的.当发生的时候,抛出异常和回滚就足
- 我们绝大部分人估计都还在用着jdk8,12其实是一个非LTS(long time support)版本,而11与8一样是LTS版,意味着下个
- 本文实例讲述了Java访问WebService返回XML数据的方法。分享给大家供大家参考。具体如下:import java.io.IOExc
- 1.用途在SpringBoot中,通过jasypt可以进行加密解密. 这个是双向的, 且可以配置密钥.2.使用:2.1通过UT创建工具类,并
- 克隆方法是原型设计模式中必须使用的方式,它将返回一个与当前对象数据一致的对象。正如其名,犹如一个模子雕刻而出。克隆类型分为两种:浅克隆、深克
- 前言作为一个服务端开发感觉一直挺排斥框架这种东西的,总觉得什么实现逻辑都帮你封装在里面了,你只需要配置这配置那个,出了问题也不知道怎么排查,
- 访问静态资源遇到的坑及解决开始是以这种结构进行的,结果页面上一篇红,访问的页面是这样的最终找出来问题,虽然每次调整路径都不对,最终查看多种方
- 面试题1:谈一下你对 Nginx 的理解Nginx 是一款自由的、开源的、高性能的 HTTP 服务器和反向代理服务器;同时也是一个 IMAP
- 一、简介 线程池类为 java.util.concurrent.ThreadPoolExecutor,常用构造方法为:ThreadPoolE
- 前言我们在 页面切换转场动画,英雄救场更有趣!介绍了 Hero 动画效果,使用 Hero 用于转场能够提供非常不错的体验。既然称之
- 本文为大家介绍了FTP上传下载队列窗口的实现方法,供大家参考,具体内容如下1、首先看一下队列窗口的界面2、看一下上传队列窗口的界面3、看一下
- 公司运维问我怎么在windows上安装模拟器,我你说你安装模拟器干什么?他说,我安装模拟器查看app的包名这些信息做统计。我顿时想,有必要这
- 代码:package com.lwj.test.proxy;import java.lang.reflect.InvocationHandl
- BeanPostProcessor 的接口定义,可以实现提供自己的实例化逻辑,依赖解析逻辑等,也可以以后在Spring容器实例化完毕,配置和
- 闲来无事想玩玩双向通信,实现类似QQ的互发消息的功能。于是乎开始学习.Net Remoting..Net Remoting 是由客户端通过R
- package com;import java.util.Arrays; public class sjf { &nbs
- 本文实例为大家分享了PopupWindow+RecyclerView实现上下滑动框功能的具体代码,供大家参考,具体内容如下1.新建一个适配器
- Thread parameterThread_t = null; private void Print_DetailForm_S
- 对于一个App的UI而言,在流畅性上的改进目标其实就是降低屏幕绘制的延迟,创建流畅和稳定的帧率以避免卡顿。在理想情况下,全部的测量、布局和绘
- public class MainActivity extends Activity { TextView tv; Ch