【问题标题】:Android Media player keeps app instance and cause a memory leakAndroid 媒体播放器保留应用程序实例并导致内存泄漏
【发布时间】:2016-01-18 05:15:39
【问题描述】:

我有一个将媒体播放器作为成员变量的活动。

我的媒体播放器是这样初始化的:

mMediaPlayer = new MediaPlayer(); 
mMediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
mMediaPlayer.setDataSource(MyActivity.this, URL); 
mMediaPlayer.prepareAsync(); 
//i set a on Prepared Listener to start playing on Prepared

一切正常,然后我重写我的活动 onStop 方法以释放媒体播放器。

if(mMediaPlayer!=null){
   if(mMediaPlayer.isPlaying()){
       mMediaPlayer.stop();
   }
   mMediaPlayer.release();
   mMediaPlayer=null;
}

但在活动停止后,我从 LeakCanary 收到了内存泄漏报告。

报告是这样的:

MyApp.Instance->

MyApp.mLoadedApk->

LoadedApk.mReceivers->

ArrayMap.mArray->

arrayObject[].[3]->

ArrayMap.mArray->

arrayObject[][0]->

MediaPlayer.mProxyReceiver->

MediaPlayer.mProxyContext->

leaks MyActivity instance.

MyApp.Instance->

MyApp.mLoadedApk->

LoadedApk.mReceivers->

ArrayMap.mArray->

arrayObject[].[3]->

ArrayMap.mArray->

arrayObject[][0]->

MediaPlayer.mProxyReceiver->

MediaPlayer.mProxyContext->

leaks MyActivity instance.

是的,我有一个扩展 Application 的 MyApp 类,并且我在静态字段中持有对 MyApp 实例的引用,但我从未在我的活动中使用该引用,我该如何解决这个泄漏?

[编辑]

这是我的 Activity 的代码:

public class PlayActivity extends ActionBarActivity {

private MediaPlayer mediaPlayer;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_play);
    getSupportActionBar().setDisplayHomeAsUpEnabled(true);
}

@Override
protected void onStop() {
    super.onStop();
    if(mediaPlayer!=null){
        if(mediaPlayer.isPlaying()){
            mediaPlayer.stop();
        }
        mediaPlayer.release();
        mediaPlayer=null;
    }
}

public void playIt(View view){
    if(mediaPlayer==null){
        mediaPlayer = new MediaPlayer();
        mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
            @Override
            public void onPrepared(MediaPlayer mp) {
                mp.start();
            }
        });


    }
    if(!mediaPlayer.isPlaying()){
        mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
        try {
            mediaPlayer.setDataSource(PlayActivity.this, Uri.parse("http://www.noiseaddicts.com/samples_1w72b820/142.mp3"));
            mediaPlayer.prepareAsync();
        } catch (IOException e) {
            e.printStackTrace();
        }

    }



   }

 }

【问题讨论】:

  • 请发布完整的课程
  • @willermo,你解决过这个问题吗?
  • @SebastianRoth 还没有,因此我的应用程序仍在泄漏内存。

标签: android memory-leaks android-mediaplayer leakcanary


【解决方案1】:

在我的测试中,我发现这发生在 HTTP 流式传输期间。

当我在目标上检查 MediaPlayer 的源代码时(在我的情况下是 API19),确实有一个 StreamProxy 实例被设置和分配,但没有在 release 方法中释放,而只是在 reset方法。

这对我有用:

if (mediaPlayer != null) {
    mediaPlayer.reset();
    mediaPlayer.release();
    mediaPlayer = null;
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-08-31
    • 2011-07-08
    • 2021-02-25
    • 1970-01-01
    相关资源
    最近更新 更多