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程序设计有所帮助。

(0)

相关推荐

  • 实例解析使用Java实现基本的音频播放器的编写要点

    Java音频播放,因为必须依赖到本地环境,所以JAVA在音频处理方面优势不大,或者说打从Java体系开发时就没太多的考虑音频播放因素,要知道最早的Java 1.1版本中,没有后来的javax.sound包,音频只能通过Applet包调取-- 遗憾的是,在图形程序开发中,我们的程序却又难免要使用到背景音乐.效果音等配合图像操作,哎,这实在是Sun大神给我们开的一个不打不小的玩笑.万幸后来Sun大神开眼,提供了javax.sound包,才解救我们于水深火热当中~ 但是继之而来的问题是,在javax.

  • Java 使用IO流实现大文件的分割与合并实例详解

    Java 使用IO流实现大文件的分割与合并 文件分割应该算一个比较实用的功能,举例子说明吧比如说:你有一个3G的文件要从一台电脑Copy到另一台电脑, 但是你的存储设备(比如SD卡)只有1G ,这个时候就可以把这个文件切割成3个1G的文件 ,分开复制, 最后把三个文件合并, 这样就解决问题了 :再比如说, 你有一个上百M的文件要上传到FTP ,但是这个FTP限制你单个文件不能超过10M 这时候也可以用文件分割的办法解决问题.既然分割了,那么在我们再次使用的时候就需要合并了,今天我们就通过Java

  • java实现音频文件播放功能

    本文实例为大家分享了java实现音频文件的播放功能的具体代码,供大家参考,具体内容如下 实现思路 1.首先获取音频文件的地址,然后通过IO流读取音频文件,加缓冲区,实现Player类的对象. 2.Player类主要用于播放器的初始化,以及通过它来实现一些音视频文件的播放,这个类需要手动去网上下载,然后添加路径到我们Eclipse的library中. 3.Player类有两种方法比较常用,play()方法和close()方法,前者用于启动音频文件,后者用于退出音频文件的播放,这两个方法我们在使用的

  • 代码分析JAVA中PCM人声音频变声处理

    项目中需要用到对PCM人声音频数据进行变声处理.苦苦挣扎了一周终于找到了纯Java实现的一套框架--TarsosDSP.功能非常强大!可以实时音频处理!当然我只用到了对文件处理.实际上逻辑是一样的 TarsosDSP的GitHub地址:https://github.com/JorenSix/TarsosDSP 将它整合至自己的项目工程. 具体Java工具类代码: /** * 变声 * @param rawPcmInputStream 原始PCM数据输入流 * @param speedFactor

  • java实现大文件分割与合并的实例代码

    复制代码 代码如下: package com.test; import java.io.BufferedReader;  import java.io.BufferedWriter;  import java.io.FileNotFoundException;  import java.io.FileReader;  import java.io.FileWriter;  import java.io.IOException;  import java.util.Collections;  im

  • Java实现文件分割和文件合并实例

    文件切割和文件合并这个问题困扰了我有一段时间了(超过一天没做粗来). 找了好多博客,本来想转载一个来的 结果找不到了.很无奈. 只好自己贴代码上了. 当然我会尽力好好写注释的. 文件切割器: import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; import java.io.RandomAccessFile; import java.util.Scanner; public c

  • Java使用IO流实现音频的剪切和拼接

    需求: 使用IO流将指定目录下的若干个音频文件的高潮部分,进行剪切,并重新拼接成一首新的音频文件 思路(以两首歌为例): 第一首歌有一个输入流对象bis1.第二首歌有一个输入流对象bis2,他们公用一条输出流对象bos(在选择构造方法的时候选择含有布尔类型参数的那个),待第一首歌剪切完成后,在此基础上追加第二首歌的"高潮部分". 实现代码: import java.io.BufferedInputStream; import java.io.BufferedOutputStream;

  • 使用javax.sound实现简单音频播放

    本文实例为大家分享了javax.sound实现简单音频播放的具体代码,供大家参考,具体内容如下 /** * @see * @author Al_assad yulinying_1994@outlook.com * @date 2016年11月17日 下午6:27:59 * @version V1.0 * Description: 简易音频播放器(只支持AU,RA,WAV) * 在不使用JMF的情况下快速实现音频播放 * */ import javax.sound.sampled.*; impor

  • java音频播放示例分享(java如何播放音频)

    这是一份精简后的代码,能够支持的格式十分有限. 复制代码 代码如下: package com.hongyuan.test; import java.io.File;import java.io.IOException; import javax.sound.sampled.AudioFormat;import javax.sound.sampled.AudioInputStream;import javax.sound.sampled.AudioSystem;import javax.sound

  • Java将文件分割为多个子文件再将子文件合并成原始文件的示例

    Java将文件分割为多个子文件再将子文件合并成原始文件的示例,废话不多说,代码如下: import java.io.File; import java.io.InputStream; import java.io.FileInputStream; import java.io.OutputStream; import java.io.FileOutputStream; import java.util.Properties; import java.util.Iterator; import j

  • java 流操作对文件的分割和合并的实例详解

    java 流操作对文件的分割和合并的实例详解 学习文件的输入输出流,自己做一个小的示例,对文件进行分割和合并. 下面是代码: package com.dufy.file; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.SequenceInputStream; import java.ut

随机推荐