Ошибка MediaPlayer setDataSource со статусом =0x80000000 для мелодии звонка, установленной путем filepath в 2.3.4

Название говорит о многом.

Мое приложение воспроизводит мелодии, на которые указывает URI, например content://media/internal/audio/media/387 или же content://media/external/audio/media/1655 (для пользовательских рингтонов на SDcard я считаю), используя оба setDataSource(fileInfo) а также setDataSource(mContext, Uri.parse(fileInfo)),

В каждом случае я получил журналы с информацией о setDataSource failed.: status=0x80000000 исключение на телефонах с использованием Android 4.x (разные версии).

Видя, что ошибка происходит только с рингтонами, указанными в содержимом URI, но не с отдельными файлами, указанными путем, я решил также использовать пути для рингтонов, которые исправили проблему на указанных выше телефонах (при этом все еще используя setDataSource(mContext, Uri.parse(fileInfo)))

Однако на телефонах с Android 2.3.4-2.3.6 возникли проблемы (но не на моем 2.3.3):

  • Я получил несколько журналов за исключением: setDataSource failed.: status=0x80000000 для файлов с путями типа /system/media/audio/ringtones/TwirlAway.ogg
  • Я также получил журнал о MediaPlayer.onErrorListener.onError(int what, int extra) вызов метода с what=1 а также extra=-2147483648, что, насколько я знаю, предполагает, что файл отсутствует или поврежден. Однако я выполняю

    File file = new File(fileInfo);
    if (!file.exists())
    

проверьте в такой ситуации и верните, что файл существует - он поврежден? Очень маловероятно для музыкального файла во внутренней памяти.

Подводить итоги:

  • работает с setDataSource("content://media/internal/audio/media/52")
  • выдает исключение: setDataSource failed.: status=0x80000000 за setDataSource(mContext, "/system/media/audio/ringtones/TwirlAway.ogg")

Интересно, что первые несколько строк setDataSource(Context context, Uri uri, Headers headers) метод, который вызывается setDataSource(Context context, Uri uri) являются ( из источника GrepCode для 2.3.4):

 String scheme = uri.getScheme();
     if(scheme == null || scheme.equals("file")) {
         setDataSource(uri.getPath());
         return;
     }

Так что, в конце концов, это просто не setDataSource("/system/media/audio/ringtones/TwirlAway.ogg"), Я взял пути к мелодиям из Uris, используя:

private static String getRingtonePathFromContentUri(Context context,
        Uri contentUri) {

    String[] proj = { MediaStore.Audio.Media.DATA };
    Cursor ringtoneCursor = context.getContentResolver().query(contentUri,
            proj, null, null, null);
    ringtoneCursor.moveToFirst();
    return ringtoneCursor.getString(ringtoneCursor
            .getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));
}

Есть идеи, что может быть причиной ошибки? Может быть, это некоторые проблемы, вызванные отсутствием разрешений на чтение? Я думаю, исходный код для встроенной функции setDataSource(String path) очень помог бы, но я не смог его найти.

5 ответов

Решение

Ответ Лорна ниже был наиболее полезен при решении этой проблемы.

Для тех, кто борется с этим, вот код, который я использую более 6 месяцев, с ошибками, о которых больше не сообщалось.

fileinfo может быть как ниже (примеры):

/system/media/audio/alarms/Walk_in_the_forest.ogg

content://media/internal/audio/media/20

public static void setMediaPlayerDataSource(Context context,
        MediaPlayer mp, String fileInfo) throws Exception {

    if (fileInfo.startsWith("content://")) {
        try {
            Uri uri = Uri.parse(fileInfo);
            fileInfo = getRingtonePathFromContentUri(context, uri);
        } catch (Exception e) {
        }
    }

    try {
        if (android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.HONEYCOMB)
            try {
                setMediaPlayerDataSourcePreHoneyComb(context, mp, fileInfo);
            } catch (Exception e) {
                setMediaPlayerDataSourcePostHoneyComb(context, mp, fileInfo);
            }
        else
            setMediaPlayerDataSourcePostHoneyComb(context, mp, fileInfo);

    } catch (Exception e) {
        try {
            setMediaPlayerDataSourceUsingFileDescriptor(context, mp,
                    fileInfo);
        } catch (Exception ee) {
            String uri = getRingtoneUriFromPath(context, fileInfo);
            mp.reset();
            mp.setDataSource(uri);
        }
    }
}

private static void setMediaPlayerDataSourcePreHoneyComb(Context context,
        MediaPlayer mp, String fileInfo) throws Exception {
    mp.reset();
    mp.setDataSource(fileInfo);
}

private static void setMediaPlayerDataSourcePostHoneyComb(Context context,
        MediaPlayer mp, String fileInfo) throws Exception {
    mp.reset();
    mp.setDataSource(context, Uri.parse(Uri.encode(fileInfo)));
}

private static void setMediaPlayerDataSourceUsingFileDescriptor(
        Context context, MediaPlayer mp, String fileInfo) throws Exception {
    File file = new File(fileInfo);
    FileInputStream inputStream = new FileInputStream(file);
    mp.reset();
    mp.setDataSource(inputStream.getFD());
    inputStream.close();
}

private static String getRingtoneUriFromPath(Context context, String path) {
    Uri ringtonesUri = MediaStore.Audio.Media.getContentUriForPath(path);
    Cursor ringtoneCursor = context.getContentResolver().query(
            ringtonesUri, null,
            MediaStore.Audio.Media.DATA + "='" + path + "'", null, null);
    ringtoneCursor.moveToFirst();

    long id = ringtoneCursor.getLong(ringtoneCursor
            .getColumnIndex(MediaStore.Audio.Media._ID));
    ringtoneCursor.close();

    if (!ringtonesUri.toString().endsWith(String.valueOf(id))) {
        return ringtonesUri + "/" + id;
    }
    return ringtonesUri.toString();
}

public static String getRingtonePathFromContentUri(Context context,
        Uri contentUri) {
    String[] proj = { MediaStore.Audio.Media.DATA };
    Cursor ringtoneCursor = context.getContentResolver().query(contentUri,
            proj, null, null, null);
    ringtoneCursor.moveToFirst();

    String path = ringtoneCursor.getString(ringtoneCursor
            .getColumnIndexOrThrow(MediaStore.Audio.Media.DATA));

    ringtoneCursor.close();
    return path;
}

Вы должны явно установить длину вашего файла. Используйте перегруженный метод:
AssetFileDescriptor afd = ctx.getAssets().openFd([your asset name]); mediaPlayer.setDataSource(afd.getFileDescriptor(), afd.getStartOffset(), afd.getLength());

Произошло изменение в поведении setDataSource(String path) из-за исправления ошибки в Android 4.1.1. В 4.1.1 или более поздней версии вам нужно использовать локальный путь (без протокола). Однако в 4.0.4 и более ранних версиях вам нужно было использовать URI (например, с протоколом file://).

Вот неполный фрагмент кода, который должен проиллюстрировать обходной путь:

// as of 4.1.1 (JELLY_BEAN) we need to use a local path (without protocol)
// on 4.0.4 and earlier we needed a URI (with file:// protocol)
final String cachedFile = android.os.Build.VERSION.SDK_INT >= 16 // android.os.Build.VERSION_CODES.JELLY_BEAN
                        ? getCacheFilePath(file)
                        : getCacheFileUri(file);


// for the purpose of this example 
// assume cacheFolder is a String and getCacheFile returns a String

public String getCacheFilePath(String file) {
    return cacheFolder + getCacheFile(file);
}

public String getCacheFileUri(String file) {
    return "file://" + cacheFolder + getCacheFile(file);
}

Это может происходить из-за формата файла, который вы пытаетесь воспроизвести или сжать. Я сжимал файл mp4, который хорошо работает, но когда я сжимал файл mov, приложение зависало, вызывая исключение setDataSource.

У меня была такая же ошибка, когда я пытался воспроизвести файл.wav. Давайте сделаем пример:

 private void start_player(){
// create raw folder inside the res folder if not present
MediaPlayer mPlayer = MediaPlayer.create(activity, R.raw.your_wave_audio_file);
        mPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
        mPlayer.setOnCompletionListener(new OnCompletionListener() {

            @Override
            public void onCompletion(MediaPlayer mp) {
                mp.release();
               // Your Stuff
            }
        });  
         mPlayer.start();
     }
    }

Я тоже получил ошибку статуса 0x80000000. В моем случае решением было перекодировать аудиофайл (например, в 16-битный PCM для файла.wav), и все работало, как ожидалось.

MediaPlayer mPlayer = MediaPlayer.create(activity, R.raw.your_wave_audio_file);
    mPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);

Вы не можете сделать это, потому что в функции create был вызван prepare, поэтому вы не можете изменить Тип аудиопотока после.

Ниже код работает нормально для меня:

sMediaPlayer = new MediaPlayer();
sMediaPlayer.setAudioStreamType(AudioManager.STREAM_RING);
AssetFileDescriptor assetFileDescriptor = context.getResources().
         openRawResourceFd(R.raw.cool_song);

 if(assetFileDescriptor == null) return;

 try {
    sMediaPlayer.setDataSource(assetFileDescriptor.getFileDescriptor(),
                               assetFileDescriptor.getStartOffset(), 
                               assetFileDescriptor.getLength());
    assetFileDescriptor.close();

    sMediaPlayer.setOnCompletionListener(new OnCompletionListener() {           
        @Override
        public void onCompletion(MediaPlayer mp) {
            if(sMediaPlayer != null){
                sMediaPlayer.release();
                sMediaPlayer = null;
            }
        }
    });

    sMediaPlayer.setOnPreparedListener(new OnPreparedListener() {               
        @Override
        public void onPrepared(MediaPlayer mp) {
            sMediaPlayer.start();                   
        }
    });
    sMediaPlayer.prepare();

} catch (IllegalArgumentException e) {
    HelpFunctions.showLog("ERROR = " + e);
} catch (IllegalStateException e) {
    HelpFunctions.showLog("ERROR = " + e);
} catch (IOException e) {
    HelpFunctions.showLog("ERROR = " + e);
}
Другие вопросы по тегам