タイトルはそれのほとんどを言います。
私のアプリケーションは、content://media/internal/audio/media/387
やcontent://media/external/audio/media/1655
(SDカードのカスタム着信音の場合)などのuriが指す着信音を、setDataSource(fileInfo)
とsetDataSource(mContext, Uri.parse(fileInfo))
の両方を使用して再生しています。
いずれの場合も、Android 4.x(異なるバージョン)を使用する電話でのsetDataSource failed.: status=0x80000000
例外に関する情報を含むログを受け取りました。
エラーがコンテンツ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
のログをいくつか受け取りました。また、what=1
とextra=-2147483648
を使用したMediaPlayer.onErrorListener.onError(int what, int extra)
メソッド呼び出しに関するログも受け取っています。しかし、私は実行します
File file = new File(fileInfo);
if (!file.exists())
そのような状況でチェックし、それはそのファイルが存在することを返しました-それはそれで壊れていますか?内部メモリの音楽ファイルにはほとんどありません。
総括する:
setDataSource("content://media/internal/audio/media/52")
で動作しますsetDataSource failed.: status=0x80000000
for 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")
に対してのみ失敗します。私はURIを使用して着信音へのパスを取得しました:
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)関数のソースコードは大いに役立つと思いますが、それを見つけることができませんでした。
この問題を処理するときに、以下のローンによる回答が最も役に立ちました。
苦労している人のために、ここで私が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;
}
Android 4.1.1のバグ修正により、setDataSource(String path)
の動作に変更がありました。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に例外が発生しました。
ファイルの長さを明示的に設定する必要があります。オーバーロードされたメソッドを使用します。AssetFileDescriptor afd = ctx.getAssets().openFd([your asset name]); mediaPlayer.setDataSource(afd.getFileDescriptor(), afd.getStartOffset(), afd.getLength());
.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();
}
}
私もstatus = 0x80000000エラーを受け取りました。私の場合、解決策はオーディオファイルを再エンコードすることで(たとえば、.wavファイルの16ビットPCMに)、すべてが期待どおりに機能しました。
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);
}