java实现切割wav音频文件的方法详解【附外部jar包下载】

 更新时间:2019年05月16日 10:58:57   作者:lwjwd  
这篇文章主要介绍了java实现切割wav音频文件的方法,结合实例形式详细分析了java切割wav音频文件的相关原理、操作技巧与注意事项,并附带外部jar包供读者下载,需要的朋友可以参考下

本文实例讲述了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算法相关内容感兴趣的读者可查看本站专题:《Java文件与目录操作技巧汇总》、《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》和《Java缓存操作技巧汇总

希望本文所述对大家java程序设计有所帮助。

相关文章

  • spring配置文件解析失败报”cvc-elt.1: 找不到元素 ''''beans'''' 的声明”异常解决

    spring配置文件解析失败报”cvc-elt.1: 找不到元素 ''''beans'''' 的声明”异常解决

    这篇文章主要给大家介绍了关于spring配置文件解析失败报”cvc-elt.1: 找不到元素 'beans' 的声明”异常的解决方法,需要的朋友可以参考下
    2020-08-08
  • SpringBoot中Redis的缓存更新策略详解

    SpringBoot中Redis的缓存更新策略详解

    这篇文章主要介绍了SpringBoot中Redis的缓存更新策略,缓存一般是为了应对高并发场景、缓解数据库读写压力,而将数据存储在读写更快的某种存储介质中(如内存),以加快读取数据的速度,需要的朋友可以参考下
    2023-08-08
  • j2ee mybatis注解@Data,@TableName,@TableField使用方式

    j2ee mybatis注解@Data,@TableName,@TableField使用方式

    这篇文章主要介绍了j2ee mybatis注解@Data,@TableName,@TableField使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-04-04
  • springboot 在ftl页面上使用shiro标签的实例代码

    springboot 在ftl页面上使用shiro标签的实例代码

    这篇文章主要介绍了springboot 在ftl页面上使用shiro标签的实例代码,通过文字说明结合实例的形式给大家介绍的非常详细,需要的朋友参考下吧
    2018-05-05
  • SSH原理及两种登录方法图文详解

    SSH原理及两种登录方法图文详解

    SSH(Secure Shell)是一套协议标准,可以用来实现两台机器之间的安全登录以及安全的数据传送,其保证数据安全的原理是非对称加密。本文通过图文并茂的形式给大家介绍了SSH原理及两种登录方法,一起看看吧
    2018-08-08
  • mybatisplus isNotNull不生效问题及解决

    mybatisplus isNotNull不生效问题及解决

    这篇文章主要介绍了mybatisplus isNotNull不生效问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • Java 不同版本的 Switch语句

    Java 不同版本的 Switch语句

    本文主要介绍了Java不同版本的Switch语句,自Java13以来,Switch表达式就被添加到Java核心库中,下面我们将介绍旧的Java Switch语句和新的Switch语句的区别,需要的朋友可以参考一下
    2022-06-06
  • SpringBoot整合SSO(single sign on)单点登录

    SpringBoot整合SSO(single sign on)单点登录

    这篇文章主要介绍了SpringBoot整合SSO(single sign on)单点登录,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • Java查找并高亮PDF文本过程解析

    Java查找并高亮PDF文本过程解析

    这篇文章主要介绍了Java查找并高亮PDF文本过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-08-08
  • SpringBoot整合MybatisPlus的教程详解

    SpringBoot整合MybatisPlus的教程详解

    这篇文章主要介绍了SpringBoot整合MybatisPlus的方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-11-11

最新评论