如何获取音频流的持续时间并从任何点继续音频流

时间:2012-01-23 13:30:41

标签: android audio-streaming android-mediaplayer

说明

我有一个音频播放器的代码。我可以通过单击进度条(在0到mediaplayer.getDuration()之间)从任何持续时间继续音频播放。它非常适合音频播放。

音频流中的

问题:

  • 当我从互联网服务器流式传输音频文件时(比如s3-bucket) 它开始正确流式传输。
  • 但是mediaPlayer.getDuration()和mediaPlayer.getCurrentPosition() 返回错误的值。在流媒体的开始 mediaPlayer.getCurrentPosition()返回5个小时。
  • 由于这个原因,我无法继续播放音频 指定的Stream持续时间(0到流持续时间)。

问题:

  1. 如何获取音频流的持续时间
  2. 如何从指定的持续时间继续音频流。对于 对于持续10分钟的文件的示例,我想从6日开始流式传输 分钟。
  3. 代码:

    public class MyAudioPlayer extends Activity 
    implements OnClickListener{
    
    
        MediaPlayer mediaPlayer = null;
        private boolean isPaused=false;
        private boolean isStop = true;
    
        String filePath = null;
        String productName = null;
    
        ImageButton btnPlay = null;
        ImageButton btnPause = null;
        ImageButton btnReset = null;
        ImageButton btnStop = null;
    
        AudioManager audioManager = null;
        SeekBar volControl = null;
        SeekBar progressControl = null;
        TextView progressText = null;
        long durationInMillis = -1;
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.ltd_audio_player);
    
            volControl = (SeekBar)findViewById(R.id.player_volume);
            progressControl = (SeekBar)findViewById(R.id.player_seekbar);
            progressText = (TextView) findViewById(R.id.player_progress_text);
    
            btnPlay = (ImageButton) findViewById(R.id.ic_player_play); 
    
            btnPause = (ImageButton) findViewById(R.id.ic_player_pause);  
    
            btnReset = (ImageButton) findViewById(R.id.ic_player_reset); 
    
            btnStop = (ImageButton) findViewById(R.id.ic_player_stop);   
    
            btnPlay.setOnClickListener(this);
            btnPause.setOnClickListener(this);
            btnReset.setOnClickListener(this);
            btnStop.setOnClickListener(this);
    
            filePath = getIntent().getExtras().getString("localPath");
    
            this.setPlayer();
            this.resetAndStartPlayer();
    
    
        }
    
        @Override
        protected void onResume() {
            super.onResume();   
            isPaused=false;
            progressText.postDelayed(onEverySecond, 1000);
        }
    
        @Override
        protected void onPause() {
            super.onPause();
    
            isPaused=true;
        }
        private void setProgressControl() {
            int maxVolume = mediaPlayer.getDuration();
            int curVolume = mediaPlayer.getCurrentPosition();
    
            progressControl.setMax(maxVolume);
            progressControl.setProgress(curVolume);
            progressControl.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
    
                @Override
                public void onProgressChanged(SeekBar seekbar, int progress, boolean fromUser) {
                    mediaPlayer.seekTo(progress);
                }
    
                @Override
                public void onStartTrackingTouch(SeekBar seekBar) {
                    // TODO Auto-generated method stub
    
                }
    
                @Override
                public void onStopTrackingTouch(SeekBar seekBar) {
                    // TODO Auto-generated method stub
    
                }
            });     
        }
        @Override
        public void onClick(View v) {
            switch(v.getId()){
            case R.id.ic_player_play:
                if(isStop==true){
                    try {
                        mediaPlayer.prepareAsync();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }else{
                    mediaPlayer.start();
                    isStop = true;
                }
                break;
            case R.id.ic_player_pause:
                mediaPlayer.pause();
                break;
            case R.id.ic_player_reset:
                mediaPlayer.seekTo(0);
                break;
            case R.id.ic_player_stop:
                isStop = true;
                progressControl.setProgress(0);
                mediaPlayer.stop();
                break;
            }
    
        }
        private void resetAndStartPlayer(){
            try {
                if(filePath!=null){
                    mediaPlayer.setDataSource(filePath);
                    mediaPlayer.prepareAsync();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        private void setPlayer(){
    
            getWindow().setFormat(PixelFormat.UNKNOWN);
            mediaPlayer = new MediaPlayer();    
    
            mediaPlayer.setOnBufferingUpdateListener(new OnBufferingUpdateListener() {
    
                @Override
                public void onBufferingUpdate(MediaPlayer mp, int percent) {
                    progressControl.setSecondaryProgress((progressControl.getMax()/100)*percent);
    
                }
            });
            mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
    
                @Override
                public void onPrepared(MediaPlayer mp) {
                    mediaPlayer.start();
                    isStop=false;
                    durationInMillis = mediaPlayer.getDuration();
                    MyAudioPlayer.this.setProgressControl();
                }
            });
            mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
        }
        @Override
        protected void onDestroy() {
            // TODO Auto-generated method stub
            mediaPlayer.release();
            super.onDestroy();
        }
        protected void setProgressText() {
            durationInMillis = mediaPlayer.getDuration();
            int curVolume = mediaPlayer.getCurrentPosition();
            long HOUR = 60*60*1000;
            if(progressText!=null){
                if(durationInMillis>HOUR){
                    progressText.setText(String.format("%1$tH:%1$tM:%1$tS", new Date(curVolume))
                            +" / "+String.format("%1$tH:%1$tM:%1$tS", new Date(durationInMillis)));
                }else{
                    progressText.setText(String.format("%1$tM:%1$tS", new Date(curVolume))
                            +" / "+String.format("%1$tM:%1$tS", new Date(durationInMillis)));
                }
            }       
        }
        private Runnable onEverySecond=new Runnable() {
            public void run() {
    
                if (mediaPlayer!=null) {
                    progressControl.setProgress(mediaPlayer.getCurrentPosition());
    
                    MyAudioPlayer.this.setProgressText();
                }
    
                if (!isPaused) {
                    progressText.postDelayed(onEverySecond, 1000);
                }
            }
        };
    }
    

    时间显示在进度条上方。

    时间:'当前持续时间'/'总持续时间'

    enter image description here

2 个答案:

答案 0 :(得分:13)

希望它可以解决您的问题。

1)音频流的持续时间和进度

我查看了您的代码,您的代码中存在重大错误以计算时间。您创建新日期(durationInMillis)。日期会增加您的地点性,即GMT + XX小时,这就是为什么您在开始流式传输时需要5个小时。您应该使用以下方法来计算currentProgress / duration。

protected void setProgressText() {

    final int HOUR = 60*60*1000;
    final int MINUTE = 60*1000;
    final int SECOND = 1000;

    int durationInMillis = mediaPlayer.getDuration();
    int curVolume = mediaPlayer.getCurrentPosition();

    int durationHour = durationInMillis/HOUR;
    int durationMint = (durationInMillis%HOUR)/MINUTE;
    int durationSec = (durationInMillis%MINUTE)/SECOND;

    int currentHour = curVolume/HOUR;
    int currentMint = (curVolume%HOUR)/MINUTE;
    int currentSec = (curVolume%MINUTE)/SECOND;

    if(durationHour>0){
        System.out.println(" 1 = "+String.format("%02d:%02d:%02d/%02d:%02d:%02d", 
                currentHour,currentMint,currentSec, durationHour,durationMint,durationSec));            
    }else{
        System.out.println(" 1 = "+String.format("%02d:%02d/%02d:%02d", 
                currentMint,currentSec, durationMint,durationSec));
    }
}

2)擦洗溪流。

MediaPlayer允许清理音频流。我已经在我的一个项目中实现了它,但这需要一些时间。从其他位置恢复音频流需要一些时间。

答案 1 :(得分:3)

第一个大问题是你的初始化顺序: 在resetAndStartPlayer中实际调用setDataSource之前调用getDuration。 如果没有数据源,MediaPlayer如何知道持续时间?

以下是如何正确地做到这一点:

  1. 在Activity.onCreate中调用MediaPlayer.setDataSource()

  2. 执行AsyncTask,在其doInBackground调用MediaPlayer.prepare()中,如果它从http流式传输需要一段时间,但在完成之后,玩家应该知道媒体的长度

  3. 在AsyncTask.onPostExecute中调用MediaPlayer.getDuration(),如果成功打开了流,它应该会成功;之后你可以调用MediaPlayer.start()
  4. 重要的事情:

    • getDuration在准备之后工作
    • 准备可能需要更长的时间,应该在其他线程中调用
    • 您也可以使用prepareAsync并避免使用AsyncTask,然后使用setOnPreparedListener回调来调用getDuration