我有一个问题,我一直在努力寻找解决方案。我的情况如下:
我有一个应用程序可以下载压缩的视频并将其解压缩到应用程序的私人文件夹,更具体地说是在子文件夹中。例如在/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。
我通过使用文件描述符、视频视图的侦听器、指示MODE_WORLD_READABLE的标志、视频视图尺寸的预先计算等找到了各种解决方案,但没有一个有积极的结果。
简而言之,我的问题是:
- 为什么我会收到那些错误,根据我在网上发现的是与视频文件编码有问题相关的错误?
- 在我的情况下,最好的东西是什么,视频视图或表面视图?
- 从应用程序的私人文件夹加载视频并能够播放视频的理想方法是什么?
谢谢。
编辑
在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。有什么想法吗?
为什么我会收到那些错误,根据我在网上发现的是与视频文件编码有问题相关的错误?
因为媒体播放引擎在自己的进程中运行,并且它无权读取您的文件。
在我的情况下,最好的东西是什么,视频视图或表面视图?
VideoView
包含SurfaceView
。您是使用VideoView
还是媒体播放器和SurfaceView
的组合取决于您。
从应用程序的私人文件夹加载视频并能够播放视频的理想方法是什么?
创建可以提供本地文件的ContentProvider
并使用提供程序Uri
而不是本地文件的Uri
,或者使用 openFileOutput()
和 MODE_WORLD_READABLE
创建本地文件。