欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java實現(xiàn)切割wav音頻文件的方法詳解【附外部jar包下載】

 更新時間:2019年05月16日 10:58:57   作者:lwjwd  
這篇文章主要介紹了java實現(xiàn)切割wav音頻文件的方法,結(jié)合實例形式詳細分析了java切割wav音頻文件的相關(guān)原理、操作技巧與注意事項,并附帶外部jar包供讀者下載,需要的朋友可以參考下

本文實例講述了java實現(xiàn)切割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字節(jié))
 * @author lwj
 *
 */
public class WavCut {
  /**
   * 截取wav音頻文件
   * @param sourcepath 源文件地址
   * @param targetpath 目標文件地址
   * @param start 截取開始時間(秒)
   * @param end 截取結(jié)束時間(秒)
   *
   * 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; //音頻數(shù)據(jù)大?。?4為128kbps比特率wav文件頭長度)
      long splitSize = (wavSize/t1)*(end-start); //截取的音頻數(shù)據(jù)大小
      long skipSize = (wavSize/t1)*start; //截取時跳過的音頻數(shù)據(jù)大小
      int splitSizeInt = Integer.parseInt(String.valueOf(splitSize));
      int skipSizeInt = Integer.parseInt(String.valueOf(skipSize));
      ByteBuffer buf1 = ByteBuffer.allocate(4); //存放文件大小,4代表一個int占用字節(jié)數(shù)
      buf1.putInt(splitSizeInt+36); //放入文件長度信息
      byte[] flen = buf1.array(); //代表文件長度
      ByteBuffer buf2 = ByteBuffer.allocate(4); //存放音頻數(shù)據(jù)大小,4代表一個int占用字節(jié)數(shù)
      buf2.putInt(splitSizeInt); //放入數(shù)據(jù)長度信息
      byte[] dlen = buf2.array(); //代表數(shù)據(jù)長度
      flen = reverse(flen); //數(shù)組反轉(zhuǎn)
      dlen = reverse(dlen);
      byte[] head = new byte[44]; //定義wav頭部信息數(shù)組
      fis.read(head, 0, head.length); //讀取源wav文件頭部信息
      for(int i=0; i<4; i++){ //4代表一個int占用字節(jié)數(shù)
        head[i+4] = flen[i]; //替換原頭部信息里的文件長度
        head[i+40] = dlen[i]; //替換原頭部信息里的數(shù)據(jù)長度
      }
      byte[] fbyte = new byte[splitSizeInt+head.length]; //存放截取的音頻數(shù)據(jù)
      for(int i=0; i<head.length; i++){ //放入修改后的頭部信息
        fbyte[i] = head[i];
      }
      byte[] skipBytes = new byte[skipSizeInt]; //存放截取時跳過的音頻數(shù)據(jù)
      fis.read(skipBytes, 0, skipBytes.length); //跳過不需要截取的數(shù)據(jù)
      fis.read(fbyte, head.length, fbyte.length-head.length); //讀取要截取的數(shù)據(jù)到目標數(shù)組
      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;
  }
  /**
  * 數(shù)組反轉(zhuǎn)
  * @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字節(jié)。

可以把頭信息作為一個對象,用ByteBuffer獲取頭信息。

注意:wave文件的頭信息字節(jié)數(shù)組中每個屬性都進行了數(shù)組反轉(zhuǎn)

wave頭信息對象模型如下:

/**
 * wave文件頭信息
 * @author lwj
 *
 */
public class Head {
  public int riff_id;      //4 byte , 'RIFF'
  public int file_size;     //4 byte , 文件長度(數(shù)據(jù)長度+36)
  public int riff_type;     //4 byte , 'WAVE'
  public int fmt_id;      //4 byte , 'fmt'
  public int fmt_size;     //4 byte , 數(shù)值為16或18,18則最后又附加信息
  public short fmt_tag;     //2 byte , 編碼方式,一般為0x0001
  public short fmt_channel;   //2 byte , 聲道數(shù)目,1--單聲道;2--雙聲道
  public int fmt_samplesPerSec;//4 byte , 采樣頻率
  public int avgBytesPerSec;  //4 byte , 每秒所需字節(jié)數(shù),記錄每秒的數(shù)據(jù)量
  public short blockAlign;   //2 byte , 數(shù)據(jù)塊對齊單位(每個采樣需要的字節(jié)數(shù))
  public short bitsPerSample;  //2 byte , 每個采樣需要的bit數(shù)
  public int data_id;      //4 byte , 字符data
  public int data_size;     //4 byte , 數(shù)據(jù)長度
  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

更多關(guān)于java算法相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java文件與目錄操作技巧匯總》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》和《Java緩存操作技巧匯總

希望本文所述對大家java程序設(shè)計有所幫助。

相關(guān)文章

  • 深入理解Java 對象和類

    深入理解Java 對象和類

    下面小編就為大家?guī)硪黄钊肜斫釰ava 對象和類。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-05-05
  • SpingMvc復雜參數(shù)傳收總結(jié)

    SpingMvc復雜參數(shù)傳收總結(jié)

    這篇文章主要為大家介紹了SpingMvc復雜參數(shù)傳收總結(jié),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-08-08
  • JPA設(shè)置默認字段及其長度詳解

    JPA設(shè)置默認字段及其長度詳解

    JPA是Java Persistence API的簡稱,中文名Java持久層API,是JDK 5.0注解或XML描述對象-關(guān)系表的映射關(guān)系,并將運行期的實體對象持久化到數(shù)據(jù)庫中。本文主要介紹了JPA如何設(shè)置默認字段及其長度,感興趣的同學可以了解一下
    2021-12-12
  • JRebel在線激活破解實現(xiàn)教程

    JRebel在線激活破解實現(xiàn)教程

    這篇文章主要介紹了JRebel在線激活破解實現(xiàn)教程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-12-12
  • 在eclipse中修改tomcat的部署路徑操作

    在eclipse中修改tomcat的部署路徑操作

    這篇文章主要介紹了在eclipse中修改tomcat的部署路徑操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01
  • 詳解Java如何使用集合來實現(xiàn)一個客戶信息管理系統(tǒng)

    詳解Java如何使用集合來實現(xiàn)一個客戶信息管理系統(tǒng)

    讀萬卷書不如行萬里路,只學書上的理論是遠遠不夠的,只有在實戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用Java 集合實現(xiàn)一個客戶信息管理系統(tǒng),大家可以在過程中查缺補漏,提升水平
    2021-11-11
  • Spring的@RequestParam對象綁定方式

    Spring的@RequestParam對象綁定方式

    這篇文章主要介紹了Spring的@RequestParam對象綁定方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • 詳解Quartz 與 Spring框架集成的三種方式

    詳解Quartz 與 Spring框架集成的三種方式

    這篇文章主要介紹了詳解Quartz 與 Spring框架集成的三種方式,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-11-11
  • Java中SpringCloud的五大組件詳解

    Java中SpringCloud的五大組件詳解

    這篇文章主要介紹了Java中SpringCloud的五大組件詳解,Spring cloud是一個基于Spring Boot實現(xiàn)的服務(wù)治理工具包,在微服務(wù)架構(gòu)中用于管理和協(xié)調(diào)服務(wù),需要的朋友可以參考下
    2023-07-07
  • 200行Java代碼編寫一個計算器程序

    200行Java代碼編寫一個計算器程序

    本篇文章給大家分享的只用200行java代碼,實現(xiàn)一個計算器程序,不僅能夠計算加減乘除,還能夠匹配小括號。實現(xiàn)代碼超簡單,需要的朋友參考下吧
    2017-12-12

最新評論