Выполните медленную задачу на большом количестве данных в 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 ответ

Решение использовать одну асинхронную задачу для всего списка или использовать список в качестве очереди для многих асинктических задач зависит от вашего желания параллелизма.

  • Если вы планируете распараллеливать получение тегов, то используйте несколько асинхронных задач (это было бы хорошо, если бы операция требовала много дискового ввода-вывода, а не процессорного времени).
  • В противном случае, если вы планируете выполнять их последовательно, вам следует использовать только одну асинхронную задачу. Нет необходимости создавать, а затем уничтожать десятки / сотни фоновых потоков. Это неэффективно.
Другие вопросы по тегам