Выполните медленную задачу на большом количестве данных в RecyclerView
Я создаю музыкальный проигрыватель, который получает тег BPM ID3v2 из всех mp3-файлов на устройстве и позволяет сортировать и фильтровать по BPM. Чтобы получить тег, я использую библиотеку mp3agic. Оказывается, что выборка тега занимает некоторое время для каждой песни, а выполнение выборки при добавлении каждой песни в мой список песен на некоторое время выводит проигрыватель из строя без функции. Я не уверен, должен ли я использовать AsyncTask во всем списке, когда он завершает сборку, или AsyncTask для каждого элемента, чтобы получить его BPM и вставить его в песню, или даже использовать какой-то другой, более быстрый метод получения значения BPM. Кто-нибудь может предложить какое-либо руководство?
Вот мой код для получения BPM:
public int getBpmFromId(long id) {
int bpm = -1;
Uri uri = Uri.withAppendedPath(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, Long.toString(id));
try {
Mp3File file = new Mp3File(getRealPathFromURI(getApplicationContext(), uri));
if(file.hasId3v2Tag()) {
ID3v2 id3v2Tag = file.getId3v2Tag();
bpm = id3v2Tag.getBPM();
Log.d("MP3AGIC", "Got BPM for track: " + id + ": " + bpm);
}
} catch (Exception e) {
e.printStackTrace();
}
return bpm;
}
и вот код, который я использую для создания своего списка песен:
public void getSongList() {
ContentResolver musicResolver = getContentResolver();
Uri musicUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
String selectionMimeType = MediaStore.Files.FileColumns.MIME_TYPE + "=?";
String selection = MediaStore.Audio.Media.IS_MUSIC + " != 0 AND " + selectionMimeType;
String sortOrder = null;
String[] projection = null;
String mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension("mp3");
String[] selectionArgsMp3 = new String[] { mimeType };
Cursor musicCursor = musicResolver.query(musicUri, projection, selection, selectionArgsMp3, sortOrder);
if(musicCursor != null && musicCursor.moveToFirst()) {
int idColumn = musicCursor.getColumnIndex(MediaStore.Audio.Media._ID);
int titleColumn = musicCursor.getColumnIndex(MediaStore.Audio.Media.TITLE);
int artistColumn = musicCursor.getColumnIndex(MediaStore.Audio.Media.ARTIST);
do {
long id = musicCursor.getLong(idColumn);
String title = musicCursor.getString(titleColumn);
String artist = musicCursor.getString(artistColumn);
//bpm processing
int bpm = getBpmFromId(id);
songList.add(new Song(id, title, artist));
} while (musicCursor.moveToNext());
}
if(musicCursor != null)
musicCursor.close();
}
РЕДАКТИРОВАТЬ: кажется, что может потребоваться так много времени, чтобы получить путь к файлу из Uri, что делается в этом методе:
public String getRealPathFromURI(Context context, Uri contentUri) {
Cursor cursor = null;
try {
String[] proj = { MediaStore.Images.Media.DATA };
cursor = context.getContentResolver().query(contentUri, proj, null, null, null);
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
} finally {
if (cursor != null) {
cursor.close();
}
}
}
Я не уверен, как, или если я могу, оптимизировать это.
1 ответ
Решение использовать одну асинхронную задачу для всего списка или использовать список в качестве очереди для многих асинктических задач зависит от вашего желания параллелизма.
- Если вы планируете распараллеливать получение тегов, то используйте несколько асинхронных задач (это было бы хорошо, если бы операция требовала много дискового ввода-вывода, а не процессорного времени).
- В противном случае, если вы планируете выполнять их последовательно, вам следует использовать только одну асинхронную задачу. Нет необходимости создавать, а затем уничтожать десятки / сотни фоновых потоков. Это неэффективно.