1

MediaPlayer で Android 4.3 (およびその下位バージョン) でビデオを確実に再生するのに問題があります。ほとんどのコードは標準的なボイラープレートです (完全を期すために含まれています)。

私は APK 拡張ライブラリを使用しています。私の obb は、いくつかのサムネイルといくつかの m4v ビデオファイルの zip と、ビデオを説明するための XML ファイルです。

エラーは小さなファイル (<30 秒または私の場合は <3Mb) を再生するときにのみ発生するように見えるため、タイミングの問題があると思います。mMediaPlayer.prepare()次のコードで実行すると発生します。

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    ...

    setContentView(R.layout.activity_play_video);
    Intent intent = getIntent();
    mFileName = intent.getStringExtra("Video"); //which video should we play
    ...
    mHolder = mSurfaceView.getHolder();
    mHolder.addCallback(this);
}


@Override
public void surfaceCreated(SurfaceHolder holder) {
            playVideo();
}


public void playVideo() {
    String videoFilePath = Const.CONTENT_URI + "/" + mFileName + ".m4v";
    Uri zipVideo = Uri.parse(videoFilePath);

    mMediaPlayer = new MediaPlayer();
    try {
        mMediaPlayer.setDataSource(getApplicationContext(), zipVideo);
        mMediaPlayer.setDisplay(mHolder);
        mMediaPlayer.prepare();
        mMediaPlayer.setOnPreparedListener(this);
        mMediaPlayer.setOnErrorListener (new MediaPlayer.OnErrorListener() {
            @Override
            public boolean onError(MediaPlayer mp, int what, int extra) {
                Log.d("ERROR", "what: " + what + " extra: " + extra);
                return false;
            }
        });
    } catch (IOException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
        finish();
    }
    ...
}

@Override
public void onPrepared(MediaPlayer mp) {
    Log.d("playback " , "onPrepared");
    start();
}

私は問題を調査しており、呼び出される前setDataSource(...)の正しい状態を設定していないようです。mMediaPlayerprepare()

この問題をさらに興味深いものにしているのは、大きなファイルを再生しても問題なく動作することです。

編集: logcat からの出力です。

11-01 13:19:12.483: E/MediaPlayer(26369): prepareAsync called in state 1
11-01 13:19:12.483: W/System.err(26369): java.lang.IllegalStateException
11-01 13:19:12.483: W/System.err(26369):    at android.media.MediaPlayer.prepare(Native Method)
11-01 13:19:12.483: W/System.err(26369):    at com.skilessons4u.lite.PlayVideoActivity.prepareVideo(PlayVideoActivity.java:126)
11-01 13:19:12.483: W/System.err(26369):    at com.skilessons4u.lite.PlayVideoActivity.surfaceCreated(PlayVideoActivity.java:100)
11-01 13:19:12.483: W/System.err(26369):    at android.view.SurfaceView.updateWindow(SurfaceView.java:571)
11-01 13:19:12.483: W/System.err(26369):    at android.view.SurfaceView.access$000(SurfaceView.java:86)
11-01 13:19:12.483: W/System.err(26369):    at android.view.SurfaceView$3.onPreDraw(SurfaceView.java:175)
11-01 13:19:12.483: W/System.err(26369):    at android.view.ViewTreeObserver.dispatchOnPreDraw(ViewTreeObserver.java:833)
11-01 13:19:12.483: W/System.err(26369):    at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1860)
11-01 13:19:12.483: W/System.err(26369):    at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1004)
11-01 13:19:12.483: W/System.err(26369):    at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5481)
11-01 13:19:12.483: W/System.err(26369):    at android.view.Choreographer$CallbackRecord.run(Choreographer.java:749)
11-01 13:19:12.483: W/System.err(26369):    at android.view.Choreographer.doCallbacks(Choreographer.java:562)
11-01 13:19:12.483: W/System.err(26369):    at android.view.Choreographer.doFrame(Choreographer.java:532)
11-01 13:19:12.483: W/System.err(26369):    at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:735)
11-01 13:19:12.483: W/System.err(26369):    at android.os.Handler.handleCallback(Handler.java:730)
11-01 13:19:12.483: W/System.err(26369):    at android.os.Handler.dispatchMessage(Handler.java:92)
11-01 13:19:12.483: W/System.err(26369):    at android.os.Looper.loop(Looper.java:137)
11-01 13:19:12.483: W/System.err(26369):    at android.app.ActivityThread.main(ActivityThread.java:5103)
11-01 13:19:12.483: W/System.err(26369):    at java.lang.reflect.Method.invokeNative(Native Method)
11-01 13:19:12.483: W/System.err(26369):    at java.lang.reflect.Method.invoke(Method.java:525)
11-01 13:19:12.483: W/System.err(26369):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
11-01 13:19:12.483: W/System.err(26369):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
11-01 13:19:12.483: W/System.err(26369):    at dalvik.system.NativeStart.main(Native Method)

完全を期すために、obb ファイルを生成するためにスクリプトで実行するコマンドをここに示します。サムネイルは問題なく表示されます。

zip -rn .m4v:.png $ZIPFILE $XMLFILE $VIDEOSDIR $THUMBDIR
zipalign 4 $ZIPFILE $OBBFILE
4

1 に答える 1

0

Uriの代わりにAssetFileDescriptorを使用することで、これを機能させることができました

最終的に私の問題を解決したアプローチは、次のことでした。

AssetFileDescriptor afd = expansionFile.getAssetFileDescriptor(mFileName + ".m4v");
FileDescriptor fd = afd.getFileDescriptor();
long offset = afd.getStartOffset();
long length = afd.getLength();

try {
    mMediaPlayer = new MediaPlayer();
    mMediaPlayer.setDataSource(fd,offset, length);
    mMediaPlayer.prepare();
...
} catch (IOException e) {
    e.printStackTrace();
} catch (IllegalStateException e) {
    e.printStackTrace();
}

以前にそのアプローチを試したことがありますが、オフセットと長さのパラメーターがなく、うまくいきませんでした。上に表示されているのは、最終的に機能するようになったことです。

以前の私の誤ったアプローチは、次のように「ZipFileContentProvider」を使用していました。

public final static String AUTHORITY = "com.example.app.provider.ZipFileContentProvider";
public final static Uri CONTENT_URI = Uri.parse("content://" + AUTHORITY);

String videoFilePath = CONTENT_URI + "/" + mFileName + ".m4v";
Uri zipVideo = Uri.parse(videoFilePath);
mMediaPlayer = new MediaPlayer();
mMediaPlayer.setDataSource(this, zipVideo);
mMediaPlayer.setDisplay(mHolder);
mMediaPlayer.prepare();

ただし、元の質問で述べたように、これは特定のインスタンスでのみ機能し、理想とはほど遠いものでした。

于 2013-11-01T13:36:31.177 に答える