4

我有一个问题,我一直在努力寻找解决方案。我的情况如下:

我有一个应用程序可以下载压缩视频并将它们解压缩到应用程序的私人文件夹,更具体地说是在子文件夹中。例如在/data/data/my.app.package.name.here/files/assets/assets-955。

在此文件夹中,视频被解压缩。解压过程成功完成,因为在模拟器上运行应用程序时我可以毫无问题地拉取和查看视频。

然后我有另一个活动正在访问此文件夹,找到视频文件并尝试打开它。此时我收到“抱歉,无法播放此视频”的错误,并带有以下错误堆栈:

01-30 17:36:17.770: D/ContentDemoActivity(6757): File: /data/data/xxxx/files/assets/assets-955/bank_2.mp4
01-30 17:36:17.830: I/MediaPlayer(6757): prepareAsync called in state 4
01-30 17:36:17.830: E/MediaPlayer(6757): error (1, -2147483648)
01-30 17:36:17.860: E/MediaPlayer(6757): Error (1,-2147483648)
01-30 17:36:17.860: D/VideoView(6757): Error: 1,-2147483648
01-30 17:36:19.370: E/MediaPlayer(6757): stop called in state 0
01-30 17:36:19.370: E/MediaPlayer(6757): error (-38, 0)
01-30 17:36:19.370: W/MediaPlayer(6757): mediaplayer went away with unhandled events

我尝试播放视频的代码非常基本:

mView = (VideoView) findViewById(R.id.videoView);

mMediaPlayer = new MediaPlayer();

mView.requestFocus();
mHolder = mView.getHolder();

Log.d(tag, "Populating content. Assets path: " + mAssetsPath);
File f = new File(mAssetsPath);
File[] files = f.listFiles();
Log.d(tag, "File: " + files[0].toString());

mView.setVideoURI(Uri.parse(files[0].toString()));

mView.setMediaController(new MediaController(this));

并且活动的布局有一个普通的 VideoView,没有什么花哨的。

最奇怪的是,出于测试目的,我使用了相同的视频,这次是从“原始”文件夹中加载它,它运行顺利,没有问题。在那种情况下,虽然我不得不加载它:

Uri video = Uri.parse("android.resource://" + getPackageName() + "/" + R.raw.bank_2);
mVideoView.setVideoURI(video);
mVideoView.start();

我会对下载的视频执行相同的操作,但 API 中似乎没有任何功能可以让我从应用程序的私有文件夹中加载视频 Uri。

我通过使用文件描述符、videoView 的侦听器、指示 MODE_WORLD_READABLE 的标志、videoView 尺寸的预计算等找到了各种解决方案,但它们都没有产生积极的结果。

简而言之,我的问题是:

  • 为什么我会收到那些根据我在网上找到的错误,这些错误与视频文件的编码问题有关?
  • 在我的情况下,VideoView 或 surfaceView 最好使用什么?
  • 哪种方法是从应用程序的私人文件夹加载视频并能够播放的理想方法?

谢谢。

编辑

在 CommonsWare 建议之后,我采用了以下实现:

File f = new File(mAssetsPath);
File[] files = f.listFiles();
Log.d(tag, "File: " + files[0].toString());

URI uri = URI.create("file://" + (files[0].toString()));
File file = new File(uri);
try {
    Log.d(tag, "1");
    ParcelFileDescriptor parcel = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_WRITE);
   Log.d(tag, "2");
    mMediaPlayer.setDataSource(parcel.getFileDescriptor());
    Log.d(tag, "3");
    mMediaPlayer.start();
    Log.d(tag, "4");
} catch (FileNotFoundException e) {
    e.printStackTrace();
} catch (IllegalArgumentException e) {
    e.printStackTrace();
} catch (IllegalStateException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

 Log.d(tag, "5");

不幸的是,这次我收到以下错误:

01-31 12:40:11.480: D/ContentDemoActivity(15896): File: /data/data/com.houseofradon.meb/files/assets/assets-955/bank_2.mp4
01-31 12:40:11.480: D/ContentDemoActivity(15896): 1
01-31 12:40:11.480: D/ContentDemoActivity(15896): 2
01-31 12:40:11.500: D/ContentDemoActivity(15896): 3
01-31 12:40:11.500: E/MediaPlayer(15896): start called in state 2
01-31 12:40:11.500: E/MediaPlayer(15896): error (-38, 0)
01-31 12:40:11.500: D/ContentDemoActivity(15896): 4
01-31 12:40:11.500: D/ContentDemoActivity(15896): 5
01-31 12:40:11.530: E/MediaPlayer(15896): Error (-38,0)

所以,当媒体播放器启动时会发生一些事情。错误代码 -38 似乎并不意味着我在这里找到的任何特定内容。

知道我错过了什么吗???

编辑#2

我现在使用 mediaPlayer 和 SurfaceView 以及 SurfaceHolder 侦听器来完成整个过程。这是代码:

mMediaPlayer = new MediaPlayer();

mSurfaceView = (SurfaceView) findViewById(R.id.surface);
mHolder = mSurfaceView.getHolder();
mHolder.addCallback(this);


public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    Log.d(tag, "surfaceChanged");

    try {
        mMediaPlayer.setDisplay(mHolder);
        Log.d(tag, "7");
        mMediaPlayer.start();
        Log.d(tag, "8");
    } catch (IllegalStateException e) {
        e.printStackTrace();
    }

    Log.d(tag, "9");
}



public void surfaceCreated(SurfaceHolder holder) {
    Log.d(tag, "surfaceCreated");
    File f = new File(mAssetsPath);
    File[] files = f.listFiles();
    Log.d(tag, "File: " + files[0].toString());

    URI uri = URI.create("file://" + (files[0].toString()));
    File file = new File(uri);
    try {
        Log.d(tag, "1");
        ParcelFileDescriptor parcel = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_WRITE);
        Log.d(tag, "2");
        mMediaPlayer.setDataSource(parcel.getFileDescriptor());
        Log.d(tag, "3");
        mMediaPlayer.setVolume(100, 100);
        Log.d(tag, "4");
        mMediaPlayer.prepare();
        Log.d(tag, "5");

    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }

    Log.d(tag, "6");
}

我可以听视频的音频,但图片只是纯黑色。我几乎在视频播放结束时也收到一个错误,上面写着:

01-31 14:26:01.300: W/AudioSystem(17165): AudioFlinger server died!
01-31 14:26:01.300: W/IMediaDeathNotifier(17165): media server died
01-31 14:26:01.300: E/MediaPlayer(17165): error (100, 0)
01-31 14:26:01.300: E/MediaPlayer(17165): Error (100,0)

我正在使用一个实际的设备,三星 Galaxy Tab 10.1。有任何想法吗 ?

4

1 回答 1

4

为什么我会收到那些根据我在网上找到的错误,这些错误与视频文件的编码问题有关?

因为媒体播放引擎运行在自己的进程中,它没有权限读取你的文件。

在我的情况下,VideoView 或 surfaceView 最好使用什么?

AVideoView包含一个SurfaceView。是使用VideoViewMediaPlayer 还是结合使用SurfaceView取决于您。

哪种方法是从应用程序的私人文件夹加载视频并能够播放的理想方法?

创建一个ContentProvider可以提供本地文件并使用提供程序Uri而不是本地文件的提供程序,或者使用andUri创建本地文件。openFileOutput()MODE_WORLD_READABLE

于 2012-01-30T18:18:02.277 回答