2

現在、ビデオを内部ストレージにダウンロードして(残念ながらSDカードを持っていません)、単に再生するアプリケーションに取り組んでいます。私がこれまでに試したことは次のとおりです。

標準の VideoView、/data/data/.../file.mp4 をパスとして設定 - 機能しませんでした。SurfaceView で MediaPlayer を使用し、パスまたはファイル記述子を使用 - 機能しませんでした。

私が今ここに持っているのは、VideoView のわずかに変更されたバージョンで、次の変更があります。

        if(mUri != null)
        {
            mMediaPlayer.setDataSource(mContext, mUri);
        }
        else
        {
            mMediaPlayer.setDataSource(mFd);
        }

また、ファイル記述子を設定できる setFD メソッドもあります。これは、MediaPlayer/SurfaceView 全体をアクティビティに入れるよりもはるかに簡単でクリーンです。問題は次のとおりです。

内部ストレージのファイルは次のように作成されます。

                FileInputStream fis = openFileInput(Downloader.TMP_FILENAME);
                FileOutputStream fos = openFileOutput(Downloader.FILENAME
                        + ext, Context.MODE_WORLD_READABLE);
                byte[] buf = new byte[2048];
                while (fis.read(buf) > 0) {
                    fos.write(buf);
                }
                fis.close();
                fos.close();
                deleteFile(Downloader.TMP_FILENAME);
                Log.d(TAG, "Replacement done!");

基本的には、一時ファイルを読み取り(現在再生されているファイルを上書きしたくないため..しかし、それは実際には問題ではありません)、新しいファイルに(onCompletion後に)書き込み、その後一時ファイルを削除します。

これまでに 3 つの異なる Android バージョンとデバイスでテストしました: - Android 2.2 を搭載したタブレット - Android 4.2 を搭載した Nexus 7 - Android 4.1.2 を搭載した HTC Sensation

それは単に機能しませんでした。ビデオを含む最初の100ミリ秒を再生します。音が出て、ポップアップがポップアップし、logcat が教えてくれます:

12-19 14:33:48.074: W/MediaPlayer(5560): info/warning (3, 0)
12-19 14:33:48.074: I/MediaPlayer(5560): Info (3,0)
12-19 14:33:48.394: E/MediaPlayer(5560): error (1, -1007)
12-19 14:33:48.394: E/MediaPlayer(5560): Error (1,-1007)

しかし、これが何を意味するのか、私にはまったくわかりません。インターネット全体で検索しましたが、見つかったのは「WORLD_READABLEを使用すると問題が解決します!」というものだけでした。または「ファイル記述子を使用するだけで、パーミッションを気にしないようです!」。しかし、私にとってはうまくいきません。

助けていただければ幸いです。

編集:

            FileInputStream fid = new FileInputStream((getFilesDir() + "/"
                    + Downloader.FILENAME + ext)));

            mVideoView.setVideoFD(fid.getFD());

これが、File Descriptor を MediaPlayer に追加する方法です。

ファイルを再エンコードした後、これが得られます。

12-19 15:06:45.664: E/MediaPlayer(7616): Unable to to create media player
12-19 15:06:45.664: W/System.err(7616): java.io.IOException: setDataSourceFD failed.: status=0x80000000
12-19 15:06:45.664: W/System.err(7616):     at android.media.MediaPlayer.setDataSource(Native Method)
12-19 15:06:45.664: W/System.err(7616):     at android.media.MediaPlayer.setDataSource(MediaPlayer.java:976)

これは、Webサーバーからダウンロードする方法です。

Log.i(TAG, "Opening remote connection to " + mFile.getHost()
                + mFile.getPath());
        URLConnection c = mFile.openConnection();
        c.connect();
        final long lastModified = c.getLastModified();
        final String mExtension;
        if (c.getHeaderField("Content-Disposition") != null) {
            final String mFilename = c
                    .getHeaderField("Content-Disposition").split("=")[1];
            Log.i("Downloader", "Filename is " + mFilename
                    + ", split length is " + mFilename.split("\\.").length);
            mExtension = mFilename.split("\\.")[mFilename
                    .split("\\.").length - 1];
        } else {
            mExtension = "mp4";
        }

        InputStream is = c.getInputStream();

        Log.i(TAG, "Creating temporary local file");
        // create local temporary file
        FileOutputStream fos = mContext.openFileOutput(TMP_FILENAME,
                Context.MODE_WORLD_READABLE);

        // start reading
        byte[] buf = new byte[BUFFER_SIZE];
        int bytesRead = 0;
        int curRead = 0;
        Log.i(TAG, "Starting download.. to " + TMP_FILENAME);
        if (mDownloadChangeListener != null)
            mDownloadChangeListener.onDownloadStart();

        while ((curRead = is.read(buf)) > -1) {
            fos.write(buf);
            bytesRead += curRead;
        }
        Log.i(TAG, "Read " + bytesRead + " bytes in total.");
        Log.i(TAG, "Download finished!");
        // end of stream, tell app to rename file.
        if (mDownloadChangeListener != null)
            mDownloadChangeListener.onDownloadFinished(TMP_FILENAME,
                    mExtension);

        is.close();
        fos.close();

ダウンロード後、onDownloadFinished リスナーで次のコードを実行します。

                FileInputStream fis = openFileInput(Downloader.TMP_FILENAME);
                FileOutputStream fos = openFileOutput(Downloader.FILENAME
                        + ext, Context.MODE_WORLD_WRITEABLE);
                byte[] buf = new byte[2048];
                while (fis.read(buf) > 0) {
                    fos.write(buf);
                }
                fis.close();
                fos.close();
                deleteFile(Downloader.TMP_FILENAME);
                Log.d(TAG, "Replacement done!");

何がうまくいかない可能性がありますか?私の唯一の他の考えは、それが内部ストレージのせいだということです..しかし、エラーメッセージは何か他のことを言っていますか?

4

2 に答える 2

2

エラーはMEDIA_ERROR_MALFORMED-1007の略です。これは、再生しようとしているファイルが、公式ドキュメントに記載されているファイル仕様に準拠していないことを意味します。受け取ったメディアに問題があるか、保存方法に問題があります。

個人的には、あなたの保存方法は私には問題ないように見えるので、ソースファイル自体に問題があると推測しています。

于 2012-12-19T13:55:38.617 に答える
2

解決策を見つけました。私が受け取ったすべてのエラーは「間違ったダウンロードされたメディアファイル」から来たため、私がテストしたすべての方法はおそらく機能しました...

while((curRead = in.read(buf)) {
   out.write(buf);
}

「、0、curRead」が欠けているだけでした-すべてが機能しています。read() は常に BUFFER_SIZE 未満を返すことができ、残りの部分などに 0 バイトを書き込むためです...

ありがとう!

于 2012-12-19T19:05:09.927 に答える