android 媒体的uri,安卓 - 从相对路径+显示名称中获取媒体的URI或ID

我尝试添加音频文件,val values = ContentValues().apply {

put(MediaStore.Audio.Media.RELATIVE_PATH, libraryPart.rootFolderRelativePath) // JDrop/1/1

put(MediaStore.Audio.Media.DISPLAY_NAME, remoteLibraryEntry.getFilename()) //12.mp3

put(MediaStore.Audio.Media.IS_PENDING, 1)

if(mimeType != null)

put(MediaStore.Audio.Media.MIME_TYPE, mimeType) // audio/mpeg3

}

val collection = MediaStore.Audio.Media

.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY)

var uri = ctx.contentResolver.insert(collection, values) // returns null for around 300/2000 files consistently

尝试插入新文件时Logcat输出如下。2020-01-24 22:27:33.724 4015-7707/? E/SQLiteDatabase: Error inserting title_key= bucket_display_name=1 owner_package_name=shio.at.jdrop parent=79657 volume_name=external_primary title_resource_uri=null _display_name=12.mp3 mime_type=audio/mpeg3 _data=/storage/emulated/0/Music/JDrop/1/1/12.mp3 title= group_id=1569 artist_id=322 is_pending=1 date_added=1579901253 album_id=2958 primary_directory=Music secondary_directory=JDrop bucket_id=687581593 media_type=2 relative_path=Music/JDrop/1/1/ from {P:30220;U:10165}

android.database.sqlite.SQLiteConstraintException: UNIQUE constraint failed: files._data (code 2067 SQLITE_CONSTRAINT_UNIQUE)

at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Native Method)

at android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:879)

at android.database.sqlite.SQLiteSession.executeForLastInsertedRowId(SQLiteSession.java:790)

at android.database.sqlite.SQLiteStatement.executeInsert(SQLiteStatement.java:88)

at android.database.sqlite.SQLiteDatabase.insertWithOnConflict(SQLiteDatabase.java:1639)

at android.database.sqlite.SQLiteDatabase.insert(SQLiteDatabase.java:1494)

at com.android.providers.media.MediaProvider.insertFile(MediaProvider.java:3050)

at com.android.providers.media.MediaProvider.insertInternal(MediaProvider.java:3452)

at com.android.providers.media.MediaProvider.insert(MediaProvider.java:3240)

at android.content.ContentProvider$Transport.insert(ContentProvider.java:325)

at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:164)

at android.os.Binder.execTransactInternal(Binder.java:1032)

at android.os.Binder.execTransact(Binder.java:1005)

files._data表示文件已经存在于MediaStore。JDrop/1/1/12.mp3上没有文件,它只是在MediaStore中,需要删除它,或者为现有OutputStream获取一个MediaStore,

我尝试使用下面的代码在ID中查询MediaStore,但是没有成功,if(uri == null) {

val id: Long = ctx.contentResolver.query(

collection,

arrayOf(BaseColumns._ID),

"${MediaStore.Audio.Media.RELATIVE_PATH}=? AND ${MediaStore.Audio.Media.DISPLAY_NAME}=?",

arrayOf(libraryPart.rootFolderRelativePath, remoteLibraryEntry.getFilename()),

null)?.use {

if (it.moveToNext())

it.getLong(it.getColumnIndex(BaseColumns._ID))

else null

} ?: return false

uri = Uri.withAppendedPath(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, id!!.toString())

}

我现在还尝试用硬编码的路径对_data进行查询,没有成功。val id: Long = ctx.contentResolver.query(

collection,

arrayOf(BaseColumns._ID),

"${MediaStore.Audio.Media.DATA}=?",

arrayOf("/storage/emulated/0/Music/JDrop/1/1/12.mp3"),

null)?.use {

if (it.moveToNext())

it.getLong(it.getColumnIndex(BaseColumns._ID))

else null

} ?: return false

获取null并返回false。(查询所有内容并查看返回的内容)

我试图对entier集合执行一个小的测试查询,class TestQueryObject(val id: Long, val relativePath: String, val displayName: String)

val results = mutableListOf()

ctx.contentResolver.query(

collection,

arrayOf(MediaStore.Audio.Media._ID, MediaStore.Audio.Media.RELATIVE_PATH, MediaStore.Audio.Media.DISPLAY_NAME),

null,

null,

null)?.use {

while (it.moveToNext()) {

results.add(TestQueryObject(

id = it.getLong(it.getColumnIndex(MediaStore.Audio.Media._ID)),

relativePath = it.getString(it.getColumnIndex(MediaStore.Audio.Media.RELATIVE_PATH)),

displayName = it.getString(it.getColumnIndex(MediaStore.Audio.Media.DISPLAY_NAME))

))

}

}

var find12 = results.find { it.displayName =="12.mp3" }

它返回2557个条目的列表,但是find12仍然是空的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android获取Uri的真实路径并转换成File对象可以使用以下代码: ```java public static File getFileFromUri(Context context, Uri uri) { String filePath = null; if (DocumentsContract.isDocumentUri(context, uri)) { // 如果是Document类型的Uri,则通过Document ID来进行解析 String documentId = DocumentsContract.getDocumentId(uri); if ("com.android.providers.media.documents".equals(uri.getAuthority())) { // MediaProvider的Uri String[] split = documentId.split(":"); String type = split[0]; Uri contentUri = null; if ("image".equals(type)) { contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; } else if ("video".equals(type)) { contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI; } else if ("audio".equals(type)) { contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI; } String selection = MediaStore.Images.Media._ID + "=?"; String[] selectionArgs = new String[]{split[1]}; filePath = getDataColumn(context, contentUri, selection, selectionArgs); } else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) { // DownloadsProvider的Uri Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(documentId)); filePath = getDataColumn(context, contentUri, null, null); } else if ("com.android.externalstorage.documents".equals(uri.getAuthority())) { // ExternalStorageProvider的Uri String[] split = documentId.split(":"); if ("primary".equalsIgnoreCase(split[0])) { filePath = Environment.getExternalStorageDirectory() + "/" + split[1]; } } } else if ("content".equalsIgnoreCase(uri.getScheme())) { // 如果是普通Content类型的Uri,则直接查询该Uri对应的数据 filePath = getDataColumn(context, uri, null, null); } else if ("file".equalsIgnoreCase(uri.getScheme())) { // 如果是File类型的Uri,则直接获取File路径 filePath = uri.getPath(); } return new File(filePath); } private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) { String[] projection = {MediaStore.Images.Media.DATA}; Cursor cursor = null; try { cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null); if (cursor != null && cursor.moveToFirst()) { int columnIndex = cursor.getColumnIndexOrThrow(projection[0]); return cursor.getString(columnIndex); } } finally { if (cursor != null) { cursor.close(); } } return null; } ``` 其,getDataColumn方法用于查询指定Uri对应的文件路径,getFileFromUri方法则将Uri转换为File对象。需要注意的是,不同类型的Uri需要进行不同的解析方式,具体可以根据代码注释进行理解。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值