我有一个问题,我一直试图找到解决方案很长一段时间。我的情况如下:
我有一个应用程序可以下载压缩视频并将其解压缩到应用程序的私人文件夹,更具体地说是在子文件夹中。例如,在 /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的尺寸等等找到了各种解决方案,但没有一个有积极的结果。
简而言之,我的问题是:
感谢。
在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似乎并不意味着我发现here的任何具体内容。
知道我错过了什么???
我现在使用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。有什么想法吗?
答案 0 :(得分:4)
为什么我会根据我在网上找到的那些与视频文件编码有问题相关的错误得到错误?
因为媒体播放引擎在其自己的进程中运行,并且它无权读取您的文件。
在我的案例中使用的最佳内容是什么,VideoView或SurfaceView?
VideoView
包含SurfaceView
。您使用VideoView
还是MediaPlayer和SurfaceView
的组合取决于您。
从应用程序的私人文件夹加载视频并能够播放视频的理想方法是什么?
创建可以提供本地文件的ContentProvider
并使用提供程序Uri
而不是Uri
到本地文件,或使用{{1}创建本地文件}和openFileOutput()
。