java实现切割wav音频文件的方法详解【附外部jar包下载】
作者:lwjwd 发布时间:2021-07-08 22:29:56
标签:java,音频文件
本文实例讲述了java实现切割wav音频文件的方法。分享给大家供大家参考,具体如下:
import it.sauronsoftware.jave.Encoder;
import it.sauronsoftware.jave.MultimediaInfo;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
/**
* wav音频文件截取工具
* (适用于比特率为128kbps的wav音频文件,此类音频文件的头部信息占用长度44字节)
* @author lwj
*
*/
public class WavCut {
/**
* 截取wav音频文件
* @param sourcepath 源文件地址
* @param targetpath 目标文件地址
* @param start 截取开始时间(秒)
* @param end 截取结束时间(秒)
*
* return 截取成功返回true,否则返回false
*/
public static boolean cut(String sourcefile, String targetfile, int start, int end) {
try{
if(!sourcefile.toLowerCase().endsWith(".wav") || !targetfile.toLowerCase().endsWith(".wav")){
return false;
}
File wav = new File(sourcefile);
if(!wav.exists()){
return false;
}
long t1 = getTimeLen(wav); //总时长(秒)
if(start<0 || end<=0 || start>=t1 || end>t1 || start>=end){
return false;
}
FileInputStream fis = new FileInputStream(wav);
long wavSize = wav.length()-44; //音频数据大小(44为128kbps比特率wav文件头长度)
long splitSize = (wavSize/t1)*(end-start); //截取的音频数据大小
long skipSize = (wavSize/t1)*start; //截取时跳过的音频数据大小
int splitSizeInt = Integer.parseInt(String.valueOf(splitSize));
int skipSizeInt = Integer.parseInt(String.valueOf(skipSize));
ByteBuffer buf1 = ByteBuffer.allocate(4); //存放文件大小,4代表一个int占用字节数
buf1.putInt(splitSizeInt+36); //放入文件长度信息
byte[] flen = buf1.array(); //代表文件长度
ByteBuffer buf2 = ByteBuffer.allocate(4); //存放音频数据大小,4代表一个int占用字节数
buf2.putInt(splitSizeInt); //放入数据长度信息
byte[] dlen = buf2.array(); //代表数据长度
flen = reverse(flen); //数组反转
dlen = reverse(dlen);
byte[] head = new byte[44]; //定义wav头部信息数组
fis.read(head, 0, head.length); //读取源wav文件头部信息
for(int i=0; i<4; i++){ //4代表一个int占用字节数
head[i+4] = flen[i]; //替换原头部信息里的文件长度
head[i+40] = dlen[i]; //替换原头部信息里的数据长度
}
byte[] fbyte = new byte[splitSizeInt+head.length]; //存放截取的音频数据
for(int i=0; i<head.length; i++){ //放入修改后的头部信息
fbyte[i] = head[i];
}
byte[] skipBytes = new byte[skipSizeInt]; //存放截取时跳过的音频数据
fis.read(skipBytes, 0, skipBytes.length); //跳过不需要截取的数据
fis.read(fbyte, head.length, fbyte.length-head.length); //读取要截取的数据到目标数组
fis.close();
File target = new File(targetfile);
if(target.exists()){ //如果目标文件已存在,则删除目标文件
target.delete();
}
FileOutputStream fos = new FileOutputStream(target);
fos.write(fbyte);
fos.flush();
fos.close();
}catch(IOException e){
e.printStackTrace();
return false;
}
return true;
}
/**
* 获取音频文件总时长
* @param filePath 文件路径
* @return
*/
public static long getTimeLen(File file){
long tlen = 0;
if(file!=null && file.exists()){
Encoder encoder = new Encoder();
try {
MultimediaInfo m = encoder.getInfo(file);
long ls = m.getDuration();
tlen = ls/1000;
} catch (Exception e) {
e.printStackTrace();
}
}
return tlen;
}
/**
* 数组反转
* @param array
*/
public static byte[] reverse(byte[] array){
byte temp;
int len=array.length;
for(int i=0;i<len/2;i++){
temp=array[i];
array[i]=array[len-1-i];
array[len-1-i]=temp;
}
return array;
}
public static void main(String[] args){
System.out.println(cut("f:\\111.wav","f:\\111-cut_0_10.wav",0,10));
System.out.println(cut("f:\\111.wav","f:\\111-cut_10_20.wav",10,20));
System.out.println(cut("f:\\111.wav","f:\\111-cut_20_28.wav",20,28));
}
}
wave类型的音频文件切割时必须注意头信息,128kbps比特率的wave文件头信息占用44字节。
可以把头信息作为一个对象,用ByteBuffer获取头信息。
注意:wave文件的头信息字节数组中每个属性都进行了数组反转
wave头信息对象模型如下:
/**
* wave文件头信息
* @author lwj
*
*/
public class Head {
public int riff_id; //4 byte , 'RIFF'
public int file_size; //4 byte , 文件长度(数据长度+36)
public int riff_type; //4 byte , 'WAVE'
public int fmt_id; //4 byte , 'fmt'
public int fmt_size; //4 byte , 数值为16或18,18则最后又附加信息
public short fmt_tag; //2 byte , 编码方式,一般为0x0001
public short fmt_channel; //2 byte , 声道数目,1--单声道;2--双声道
public int fmt_samplesPerSec;//4 byte , 采样频率
public int avgBytesPerSec; //4 byte , 每秒所需字节数,记录每秒的数据量
public short blockAlign; //2 byte , 数据块对齐单位(每个采样需要的字节数)
public short bitsPerSample; //2 byte , 每个采样需要的bit数
public int data_id; //4 byte , 字符data
public int data_size; //4 byte , 数据长度
public int getRiff_id() {
return riff_id;
}
public void setRiff_id(int riff_id) {
this.riff_id = riff_id;
}
public int getFile_size() {
return file_size;
}
public void setFile_size(int file_size) {
this.file_size = file_size;
}
public int getRiff_type() {
return riff_type;
}
public void setRiff_type(int riff_type) {
this.riff_type = riff_type;
}
public int getFmt_id() {
return fmt_id;
}
public void setFmt_id(int fmt_id) {
this.fmt_id = fmt_id;
}
public int getFmt_size() {
return fmt_size;
}
public void setFmt_size(int fmt_size) {
this.fmt_size = fmt_size;
}
public short getFmt_tag() {
return fmt_tag;
}
public void setFmt_tag(short fmt_tag) {
this.fmt_tag = fmt_tag;
}
public short getFmt_channel() {
return fmt_channel;
}
public void setFmt_channel(short fmt_channel) {
this.fmt_channel = fmt_channel;
}
public int getFmt_samplesPerSec() {
return fmt_samplesPerSec;
}
public void setFmt_samplesPerSec(int fmt_samplesPerSec) {
this.fmt_samplesPerSec = fmt_samplesPerSec;
}
public int getAvgBytesPerSec() {
return avgBytesPerSec;
}
public void setAvgBytesPerSec(int avgBytesPerSec) {
this.avgBytesPerSec = avgBytesPerSec;
}
public short getBlockAlign() {
return blockAlign;
}
public void setBlockAlign(short blockAlign) {
this.blockAlign = blockAlign;
}
public short getBitsPerSample() {
return bitsPerSample;
}
public void setBitsPerSample(short bitsPerSample) {
this.bitsPerSample = bitsPerSample;
}
public int getData_id() {
return data_id;
}
public void setData_id(int data_id) {
this.data_id = data_id;
}
public int getData_size() {
return data_size;
}
public void setData_size(int data_size) {
this.data_size = data_size;
}
}
附件为wave切割程序所依赖的外部jar包: jave-1.0.2
希望本文所述对大家java程序设计有所帮助。
来源:https://lwjwd.iteye.com/blog/2025072


猜你喜欢
- 理解C#中的闭包1、 闭包的含义首先闭包并不是针对某一特定语言的概念,而是一个通用的概念。除了在各个支持函数式编程的语言中,我们会接触到它。
- 最近看了一些淘宝购物车的demo,于是也写了一个。效果图如下:主要代码如下: actvity中的代码:public class Shoppi
- 声明式事务回顾事务事务在项目开发过程非常重要,涉及到数据的一致性的问题,不容马虎!事务管理是企业级应用程序开发中必备技术,用来确保数据的完整
- 环境:主机:WIN10开发环境:Android Studio 2.2 Preview 3说明:两种方法实现底部弹出的对话框:DialogDi
- Android PopWindow 设置背景亮度的实例设置背景 /** * 设置添加屏幕的背景透明度 * @param bgAl
- Java String类的concat方法在了解concat()之前,首先需要明确的是String的两点特殊性长度不可变值不可变这两点从源码
- 概述最关键的区别是AlertDialog不能指定显示位置,只能默认显示在屏幕最中间(当然也可以通过设置WindowManager参数来改变位
- 一、Kt调用-Java参数非null的处理@NotNullclass TestJava { public void
- 前言接着上一篇,上篇已经测试通过,成功添加了数据。那么这篇主要是继续上一个项目,将项目进行工程化包装,增加一些必要配置,并且生成增删改查接口
- javax.persistence 介绍Spring Data JPA 采用约定大于配置的思想,默认了很多东西JPA是存储业务实体关联的实体
- 一、return语句执行顺序finally语句是在return语句执行之后,return语句返回之前执行的package exception
- 对于之前最火的无外乎集五福了,而五福除了加十个好友获得外,最直接的途径就是支付宝的咻一咻了。那么咻一咻具体有哪些实现方式呢?下面我们将一一介
- 本文实例讲述了C#实现DataTable映射成Model的方法。分享给大家供大家参考,具体如下:这是数据库开发中经常遇到的问题,当然,这可以
- 本文实例讲述了Android编程实现TextView部分颜色变动的方法。分享给大家供大家参考,具体如下:public class Strin
- 前言本文是精讲RestTemplate第8篇,前篇的blog访问地址如下:RestTemplate在Spring或非Spring环境下使用精
- java 实现截取字符串并按字节分别输出实例代码前言:请编写一个截取字符串的函数,输入为一个字符串和字节数,输出为按字节截取的字符串。但是要
- 一般来说,在更新DataTable或是DataSet时,如果不采用SqlParameter,那么当输入的Sql语句出现歧义时,如字符串中含有
- 使用RateLimiter通过AOP方式进行限流1、引入依赖<!-- guava 限流 --><dependency>
- AbstractQueuedSynchronizerAbstractQueuedSynchronizer 简称 AQS ,抽象队列同步器,用
- Springboot自带定时任务实现动态配置Cron参数同学们,我今天分享一下SpringBoot动态配置Cron参数。场景是这样子的:后台