AudioTrack 播放wav音频文件_audioformat.encoding.pcm_float不生效-程序员宅基地

技术标签: 音视频  android  Android音视频  

我们要想对wav文件格式操作,我们就要了解wav的文件格式

https://blog.csdn.net/qq_15255121/article/details/115168456

 通过上面我们可以知道

第8到11字节 代表当前是wave格式也就是wav格式

第20-21字节 代表当前的音频数据是什么格式 如果是1代表是pcm格式

第24-28字节,代表当前的采样率

第34-35字节,代表当前的采样大小(位深)

第44字节开始,是我们真实的数据

通过上面的分析我们可以知道,wav只是把pcm包装了一下,我们可以从头中得到采样率 采样大小  位深等参数。并没有对音频数据进行压缩。

我们再结合

https://blog.csdn.net/qq_15255121/article/details/118964421?spm=1001.2014.3001.5501 就可以实现对wav文件格式的播放

下面是完整的源代码

package com.yuanxuzhen.androidmedia.audio;

import android.media.AudioAttributes;
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.os.Build;
import android.text.TextUtils;
import android.util.Log;

import androidx.annotation.RequiresApi;

import java.io.DataInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/******
 * wav格式可以参考https://blog.csdn.net/qq_15255121/article/details/115168456
 *
 * ****/

public class WavPlayer {

    private static volatile WavPlayer instance;
    private AudioTrack mAudioTrack;
    private int audioFormat;
    private int buffersize;
    private ExecutorService executorService;
    private DataInputStream inputStream;
    private static final  int HEADER_SIZE = 44;
    private static final  int AUDIO_FORMAT_POSITION = 20;
    private static final  int AUDIO_NUM_CHANNEL_POSITION = 22;
    private static final  int AUDIO_BIT_PERSAMPLE_POSITION = 34;
    private static final  int AUDIO_SAMPLE_RATE_POSITION = 24;

    public class WavData{
        int channelMask = AudioFormat.CHANNEL_OUT_MONO;
        int sampleRate = 48000;
        int audioFormatEncodeing = AudioFormat.ENCODING_PCM_FLOAT;
    }

    private WavPlayer(){

    }

    public static WavPlayer getInstance(){
        if(instance == null){
            synchronized (WavPlayer.class){
                if(instance == null){
                    instance = new WavPlayer();
                }
            }
        }
        return instance;
    }



    public WavData getInfo(String path){
        WavData info = new WavData();
        try{
            int bitPerSample = 0;
            if(TextUtils.isEmpty(path)){
                return null;
            }
            File file = new File(path);
            if(!file.exists()){
                return null;
            }
            InputStream  wavInputStrem = new FileInputStream(file);
            /*wav头部总共占44个字节*/
            ByteBuffer buffer = ByteBuffer.allocate(HEADER_SIZE);
            buffer.order(ByteOrder.LITTLE_ENDIAN); //格式 采样率 采样大小 声道数都是小端存储

            wavInputStrem.read(buffer.array(), buffer.arrayOffset(), buffer.capacity());
            buffer.rewind();
            buffer.position(AUDIO_FORMAT_POSITION);
            int format = buffer.getShort();
            Log.e("WavPlayer", "format = " + format);
            /*pcm播放*/
            if(format != 1){
                Log.e("WavPlayer", "format is not pcm");
                return null;
            }

            buffer.rewind();
            buffer.position(AUDIO_NUM_CHANNEL_POSITION);
            int channel = buffer.getShort();
            Log.e("WavPlayer", "channel="+channel);

            if(channel == 2){
                info.channelMask = AudioFormat.CHANNEL_OUT_STEREO;
            }


            buffer.rewind();
            buffer.position(AUDIO_SAMPLE_RATE_POSITION);
            info.sampleRate = buffer.getInt();
            Log.e("WavPlayer", "sampleRate="+info.sampleRate);


            buffer.rewind();
            buffer.position(AUDIO_BIT_PERSAMPLE_POSITION);
            bitPerSample = buffer.getShort();
            Log.e("WavPlayer", "bitPerSample="+bitPerSample);
            if(bitPerSample == 16){
                info.audioFormatEncodeing = AudioFormat.ENCODING_PCM_16BIT;
            }else if(bitPerSample == 8){
                info.audioFormatEncodeing = AudioFormat.ENCODING_PCM_8BIT;
            }else{
                info.audioFormatEncodeing = AudioFormat.ENCODING_PCM_FLOAT;
            }

            wavInputStrem.close();
        }catch (Exception e){
            e.printStackTrace();
        }
        return info;
    }



    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    private void init(WavData data) {
        executorService = Executors.newCachedThreadPool();
        AudioFormat.Builder builder = new AudioFormat.Builder();
        builder.setChannelMask(data.channelMask)
                .setEncoding(data.audioFormatEncodeing)
                .setSampleRate(data.sampleRate);
        AudioFormat audioFormat = builder.build();

        buffersize = AudioTrack.getMinBufferSize(data.sampleRate,
                data.channelMask,
                data.audioFormatEncodeing);
        Log.e("yuanAudioTrack", "init buffersize=" + buffersize);
        AudioAttributes audioAttributes = (new AudioAttributes.Builder()).setContentType(AudioAttributes.CONTENT_TYPE_MUSIC).build();
        mAudioTrack = new AudioTrack(
                audioAttributes,
                audioFormat,
                buffersize,
                AudioTrack.MODE_STREAM,
                AudioManager.AUDIO_SESSION_ID_GENERATE
        );

    }


    /**
     * 播放线程
     */
    Runnable recordRunnable = new Runnable() {
        @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
        @Override
        public void run() {
            try {
                //设置线程的优先级
                android.os.Process.setThreadPriority(android.os.Process.THREAD_PRIORITY_URGENT_AUDIO);
                byte[] tempBuffer = new byte[buffersize];
                int readCount = 0;
                while (inputStream.available() > 0) {
                    readCount= inputStream.read(tempBuffer);
                    if (readCount == AudioTrack.ERROR_INVALID_OPERATION || readCount == AudioTrack.ERROR_BAD_VALUE) {
                        continue;
                    }
                    if (readCount != 0 && readCount != -1) {//一边播放一边写入语音数据
                        //判断AudioTrack未初始化,停止播放的时候释放了,状态就为STATE_UNINITIALIZED
                        if(mAudioTrack.getState() == mAudioTrack.STATE_UNINITIALIZED){
                           return;
                        }
                        mAudioTrack.play();
                        mAudioTrack.write(tempBuffer, 0, readCount);
                    }
                }
                stopPlay();//播放完就停止播放
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

    };


    private void setPath(String path) throws Exception {
        File file = new File(path);
        inputStream = new DataInputStream(new FileInputStream(file));
        inputStream.skipBytes(HEADER_SIZE); //跳过WAV头部44个字节
    }

    public void stopPlay() {
        try {
            if (mAudioTrack != null) {
                if (mAudioTrack.getState() == AudioRecord.STATE_INITIALIZED) {//初始化成功
                    mAudioTrack.stop();//停止播放
                }
                if (mAudioTrack != null) {
                    mAudioTrack.release();//释放audioTrack资源
                }
                mAudioTrack = null;
            }
            if (inputStream != null) {
                inputStream.close();//关闭数据输入流
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public void startPlay(String path) {
        try {
            WavData wavData = getInfo(path);
            if(wavData == null){
                return;
            }
            init(wavData);
            setPath(path);
            executorService.execute(recordRunnable);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }



}

gitee地址

https://gitee.com/creat151/android-media.git

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_15255121/article/details/118997052

智能推荐

while循环&CPU占用率高问题深入分析与解决方案_main函数使用while(1)循环cpu占用99-程序员宅基地

文章浏览阅读3.8k次,点赞9次,收藏28次。直接上一个工作中碰到的问题,另外一个系统开启多线程调用我这边的接口,然后我这边会开启多线程批量查询第三方接口并且返回给调用方。使用的是两三年前别人遗留下来的方法,放到线上后发现确实是可以正常取到结果,但是一旦调用,CPU占用就直接100%(部署环境是win server服务器)。因此查看了下相关的老代码并使用JProfiler查看发现是在某个while循环的时候有问题。具体项目代码就不贴了,类似于下面这段代码。​​​​​​while(flag) {//your code;}这里的flag._main函数使用while(1)循环cpu占用99

【无标题】jetbrains idea shift f6不生效_idea shift +f6快捷键不生效-程序员宅基地

文章浏览阅读347次。idea shift f6 快捷键无效_idea shift +f6快捷键不生效

node.js学习笔记之Node中的核心模块_node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是-程序员宅基地

文章浏览阅读135次。Ecmacript 中没有DOM 和 BOM核心模块Node为JavaScript提供了很多服务器级别,这些API绝大多数都被包装到了一个具名和核心模块中了,例如文件操作的 fs 核心模块 ,http服务构建的http 模块 path 路径操作模块 os 操作系统信息模块// 用来获取机器信息的var os = require('os')// 用来操作路径的var path = require('path')// 获取当前机器的 CPU 信息console.log(os.cpus._node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是

数学建模【SPSS 下载-安装、方差分析与回归分析的SPSS实现(软件概述、方差分析、回归分析)】_化工数学模型数据回归软件-程序员宅基地

文章浏览阅读10w+次,点赞435次,收藏3.4k次。SPSS 22 下载安装过程7.6 方差分析与回归分析的SPSS实现7.6.1 SPSS软件概述1 SPSS版本与安装2 SPSS界面3 SPSS特点4 SPSS数据7.6.2 SPSS与方差分析1 单因素方差分析2 双因素方差分析7.6.3 SPSS与回归分析SPSS回归分析过程牙膏价格问题的回归分析_化工数学模型数据回归软件

利用hutool实现邮件发送功能_hutool发送邮件-程序员宅基地

文章浏览阅读7.5k次。如何利用hutool工具包实现邮件发送功能呢?1、首先引入hutool依赖<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.7.19</version></dependency>2、编写邮件发送工具类package com.pc.c..._hutool发送邮件

docker安装elasticsearch,elasticsearch-head,kibana,ik分词器_docker安装kibana连接elasticsearch并且elasticsearch有密码-程序员宅基地

文章浏览阅读867次,点赞2次,收藏2次。docker安装elasticsearch,elasticsearch-head,kibana,ik分词器安装方式基本有两种,一种是pull的方式,一种是Dockerfile的方式,由于pull的方式pull下来后还需配置许多东西且不便于复用,个人比较喜欢使用Dockerfile的方式所有docker支持的镜像基本都在https://hub.docker.com/docker的官网上能找到合..._docker安装kibana连接elasticsearch并且elasticsearch有密码

随便推点

Python 攻克移动开发失败!_beeware-程序员宅基地

文章浏览阅读1.3w次,点赞57次,收藏92次。整理 | 郑丽媛出品 | CSDN(ID:CSDNnews)近年来,随着机器学习的兴起,有一门编程语言逐渐变得火热——Python。得益于其针对机器学习提供了大量开源框架和第三方模块,内置..._beeware

Swift4.0_Timer 的基本使用_swift timer 暂停-程序员宅基地

文章浏览阅读7.9k次。//// ViewController.swift// Day_10_Timer//// Created by dongqiangfei on 2018/10/15.// Copyright 2018年 飞飞. All rights reserved.//import UIKitclass ViewController: UIViewController { ..._swift timer 暂停

元素三大等待-程序员宅基地

文章浏览阅读986次,点赞2次,收藏2次。1.硬性等待让当前线程暂停执行,应用场景:代码执行速度太快了,但是UI元素没有立马加载出来,造成两者不同步,这时候就可以让代码等待一下,再去执行找元素的动作线程休眠,强制等待 Thread.sleep(long mills)package com.example.demo;import org.junit.jupiter.api.Test;import org.openqa.selenium.By;import org.openqa.selenium.firefox.Firefox.._元素三大等待

Java软件工程师职位分析_java岗位分析-程序员宅基地

文章浏览阅读3k次,点赞4次,收藏14次。Java软件工程师职位分析_java岗位分析

Java:Unreachable code的解决方法_java unreachable code-程序员宅基地

文章浏览阅读2k次。Java:Unreachable code的解决方法_java unreachable code

标签data-*自定义属性值和根据data属性值查找对应标签_如何根据data-*属性获取对应的标签对象-程序员宅基地

文章浏览阅读1w次。1、html中设置标签data-*的值 标题 11111 222222、点击获取当前标签的data-url的值$('dd').on('click', function() { var urlVal = $(this).data('ur_如何根据data-*属性获取对应的标签对象

推荐文章

热门文章

相关标签