如何在RecyclerView

时间:2017-12-05 09:30:09

标签: android android-asynctask android-recyclerview

我有一个Music类需要将其封面艺术作为位图返回,以便在RecylerView上使用它。我在类中使用AsyncTask内部类来执行检索,但是,一旦从封面艺术创建列表,我的应用程序就会冻结。请参阅下面的Music.java代码:

public class Music {
    private static final String LOG_TAG = Music.class.getSimpleName();
    private String mId;
    private String mTitle;
    private String mUrl;
    private Bitmap mCoverArt;

    public Music(String id, String title, String url) {
        mId = id;
        mTitle = title;
        mUrl = url;
        mCoverArt = null; //Initialize with null
    }

    String getId() {
        return mId;
    }

    String getTitle() {
        return mTitle;
    }

    String getUrl() {
        return mUrl;
    }

    Bitmap getCoverArt() {
        if(mCoverArt != null) {
            return mCoverArt;
        }
        else {
            Bitmap bmp = null;
            try {
                bmp = new GetCoverArt().execute(mUrl).get();
            } catch (InterruptedException e) {
                Log.e(LOG_TAG, "InterruptedException: " + e.getMessage());
            } catch (ExecutionException e) {
                Log.e(LOG_TAG, "ExecutionException: " + e.getMessage());
            }
            return bmp;

        }
    }
    public void setCoverArt(Bitmap bmp) { mCoverArt = bmp; }

    private static class GetCoverArt extends AsyncTask<String, Void, Bitmap> {
        @Override
        protected Bitmap doInBackground(String... paths) {
            MediaMetadataRetriever mmr = new MediaMetadataRetriever();
            mmr.setDataSource(paths[0], new HashMap<String,String>());
            byte[] picData = mmr.getEmbeddedPicture();
            return BitmapFactory.decodeByteArray(picData, 0, picData.length);
        }
    }
}

我通过这种方式在onBindViewHolder中为我的RecyclerView调用getCoverArt():

@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    Music song = mDataset.get(position);
    Bitmap songCoverArt = song.getCoverArt();
    String songTitle = song.getTitle();
    String songId = song.getId();
    String songUrl = song.getUrl();

    if(songCoverArt != null) {
        Glide.with(mContext).load(songCoverArt).into(holder.coverArt);
    }
    else {
        holder.coverArt.setImageResource(R.drawable.coverart_fallback);
    }
    Bitmap bmp = song.getCoverArt();
    if(bmp != null) {
        Glide.with(mContext).load(bmp).into(holder.coverArt);
    }
    else {
        Glide.with(mContext).load(R.drawable.coverart_fallback).into(holder.coverArt);
    }

我不明白为什么AsyncTask中的doInBackground可能导致UI线程冻结。我认为这一切都在后台运行,但似乎我的RecyclerView正在等待它完成工作才能使用返回的值。目前,作为一个糟糕的解决方法,当我构建Music对象并将它们添加到ArrayList时,我正在主活动中的另一个AsyncTask以及其他网络操作中进行此类处理:

 for( int j = 0 ; j < songs.length() ; j++) {
                            JSONObject song = songs.getJSONObject(j); //get song at index j
                            String songId = song.getString( getString(R.string.json_song_id) );
                            String title = song.getString( getString(R.string.json_song_title));
                            String path = song.getString( getString(R.string.json_filepath) );
                            //Create a temp Music object to extract Music info
                            Music songObj = new Music(songId, title, path);
                            Bitmap bmp = createCoverArtBmp(path);
                            songObj.setCoverArt(bmp);
                            mMusicTrackArray.add(songObj); //Add the music object into array
                        }

2 个答案:

答案 0 :(得分:3)

您无需将其从网址转换为位图以显示图片。只需将图片网址传递给Glide,它就会为您加载。

通过调用getCoverArt()中的位图制作,您不是在执行异步,而是等待任务完成,结果冻结您的视图。 请在此处查看如何将Glide与网址一起使用:https://github.com/bumptech/glide#how-do-i-use-glide

答案 1 :(得分:1)

@Umar Hussain传递网址是正确的,但Glide也可以使用本地文件或URI这样做包含在:

Glide load local image by Uri.

使用本地文件(我建议保存到缓存)的好处是你不必传递庞大的位图。内存不足导致开发人员伤心。

我注意到你也在使用if语句的后备,但是Glide有一个占位符方法

Glide.with(getContext())
                    .load(some_bitmap_file_url_or_drawable)
                    .placeholder(some_placeholder_drawable)
                    .into(view_you_want_it_to_appear);

这应该转换为加载时所需的图像,如果不是

,则提供后备图像