Производительность ListView слишком низкая, хотя внутри AsyncTask

Я занимаюсь разработкой музыкального плеера. поэтому мне нужно загрузить список песен в пользовательском ListView, который имеет один ImageView и два textView. Его производительность слишком низкая, поэтому я искал в Google и нашел решение, которое гласит: "используйте AsyncTasks и загрузите listView в фоновом режиме". Но все же производительность listview слишком медленная. Пожалуйста, помогите мне увеличить производительность listview. Вот мой код:

Этот код загружает пользовательский ArrayList, получая значения из song_database, и он находится внутри класса Activity.

private ArrayList getListData() {
    ArrayList results = new ArrayList();
    Song Data = new Song();
    Database_Song db = new Database_Song(this);
    String data = db.getData();
        StringTokenizer token = new StringTokenizer(data,";");

        while(token.hasMoreTokens())
        {
            String dataPart = token.nextToken();
            StringTokenizer tokenPart = new StringTokenizer(dataPart,"|");
            Data = new Song();
            Data.setSongTitle(tokenPart.nextToken());
            mediaInfo.setDataSource(tokenPart.nextToken());

            if (mediaInfo.getEmbeddedPicture() != null){
                byte[] img = mediaInfo.getEmbeddedPicture();
                Data.setImage((BitmapFactory.decodeByteArray(img, 0,img.length)));
            }
            else
                Data.setImage(BitmapFactory.decodeResource(this.getResources(), R.drawable.album_art));

            Data.setAlbumName(tokenPart.nextToken());
            results.add(Data);
        }

        return results;
}

Это мой класс AsyncTasks, который является внутренним классом класса Activity.

private class LoadListTask extends AsyncTask<String, Void, Integer> {
    ArrayList Song_details;
    protected void onPreExecute() {

    }

    protected Integer doInBackground(String... params) {
        Song_details = getListData();
        return 0;
    }

    protected void onPostExecute(Integer result) {
        adapter = new CustomListAdapter_Song(cont, Song_details);
        lv_song.setAdapter(adapter);

        if (result == 0) {
            adapter.notifyDataSetChanged();
         }
    }
}

это мой класс адаптера

public class CustomListAdapter_Song extends BaseAdapter{

private ArrayList<Song> listData;
private LayoutInflater layoutInflater;
private ArrayList<Song> arraylist;
private int lastPosition = -1;

public CustomListAdapter_Song(Context context, ArrayList<Song> listData) {
    this.listData = listData;
    layoutInflater = LayoutInflater.from(context);
    this.arraylist = new ArrayList<Song>();
    this.arraylist.addAll(listData);
}

@Override
public int getCount() {
    return listData.size();
}

@Override
public Object getItem(int position) {
    return listData.get(position);
}

@Override
public long getItemId(int position) {
    return position;
}

public View getView(int position, View convertView, ViewGroup parent) {
    ViewHolder holder;
    if (convertView == null) {
        convertView = layoutInflater.inflate(R.layout.listview_item_row_for_data, null);
        holder = new ViewHolder();
        holder.songName = (TextView) convertView.findViewById(R.id.Song_songTitle);
        holder.albumName = (TextView) convertView.findViewById(R.id.Song_AlbumName);
        holder.songImage = (ImageView) convertView.findViewById(R.id.Song_AlbumImage);
        convertView.setTag(holder);

    } else {
        holder = (ViewHolder) convertView.getTag();
    }

    holder.songName.setText(listData.get(position).getSongTitle());
    holder.albumName.setText(listData.get(position).getAlbumName());
    holder.songImage.setImageBitmap((listData.get(position).getImage()));
    View view = convertView;
    Animation animation = AnimationUtils.loadAnimation(view.getContext(), (position > lastPosition) ? R.anim.up_from_bottom : R.anim.down_from_top);
    view.startAnimation(animation);
    lastPosition = position;

    return convertView;
}

static class ViewHolder {
    TextView songName;
    TextView albumName;
    ImageView songImage;
}   

}

2 ответа

Не используйте AsyncTask, но вам нужен CursorLoader

public class SongsFragment extends ListFragment implements LoaderManager.LoaderCallbacks<Cursor>
{
    SongAdapter adapter;

    @Override
    public View onCreateView( LayoutInflater inflater , ViewGroup container , Bundle savedInstanceState )
    {
        return inflater.inflate(R.layout.fragment_songs, container, false);
    }

    @Override
    public void onCreate( Bundle savedInstanceState )
    {
        super.onCreate(savedInstanceState);
        adapter = new SongAdapter(getActivity(), null);
        setListAdapter(adapter);
        getLoaderManager().initLoader(0, null, this);

    }

    private class SongAdapter extends CursorAdapter
    {
        Uri sArtworkUri = Uri.parse("content://media/external/audio/albumart");

        public SongAdapter( Context context , Cursor c )
        {
            super(context , c , true);
        }

        @Override
        public View getView( int position , View convertView , ViewGroup parent )
        {
            View v;
            Cursor cursor = (Cursor) getItem(position);
            if (convertView == null)
            {
                v = newView(mContext, cursor, parent);
            }
            else
            {
                v = convertView;
            }
            bindView(v, mContext, cursor);
            return v;
        }

        @Override
        public void bindView( View view , Context context , Cursor cursor )
        {
            ViewHolder holder = (ViewHolder) view.getTag();
            holder.songNameTextView.setText(cursor.getString(2));
            holder.artistTextView.setText(cursor.getString(1));
            Uri albumArtUri = ContentUris.withAppendedId(sArtworkUri, cursor.getLong(3));
            Picasso.with(getActivity()).load(albumArtUri).error(R.drawable.placeholder).placeholder(R.drawable.placeholder).into(holder.albumImageView);
        }

        @Override
        public View newView( Context context , Cursor cursor , ViewGroup parent )
        {
            View view = LayoutInflater.from(context).inflate(R.layout.row_song, parent, false);
            ViewHolder holder = new ViewHolder();
            holder.albumImageView = (ImageView) view.findViewById(R.id.albumImg);
            holder.songNameTextView = (TextView) view.findViewById(R.id.songNameTxtView);
            holder.artistTextView = (TextView) view.findViewById(R.id.artistNameTxtView);
            view.setTag(holder);
            return view;
        }
    }

    private static class ViewHolder
    {
        public ImageView    albumImageView;
        public TextView     artistTextView;
        public TextView     songNameTextView;
    }

    @Override
    public Loader<Cursor> onCreateLoader( int arg0 , Bundle arg1 )
    {
        String[] projection = { MediaStore.Audio.Media._ID, MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.ALBUM_ID };
        String selection = MediaStore.Audio.Media.IS_MUSIC + " != 0";
        return new CursorLoader(getActivity(), MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, projection, selection, null, null);
    }

    @Override
    public void onLoadFinished( Loader<Cursor> arg0 , Cursor data )
    {
        adapter.swapCursor(data);
    }

    @Override
    public void onLoaderReset( Loader<Cursor> arg0 )
    {
        adapter.swapCursor(null);
    }
}

Идея использовать AsyncTask для загрузки данных для просмотра списка означает загрузку данных ТОЛЬКО для определенных элементов. Как вы, вероятно, знаете, ListView запрашивает у своего адаптера только те элементы, которые видны на экране (плюс пару над и под видимой областью). Как только ваш адаптер не должен предоставлять все данные сразу, вы можете загрузить их по требованию.
Ниже приведен лишь пример возможной реализации:

    // We need a queue to run only one AsyncTask
    // How to deal with queue you may find here 
    // http://developer.android.com/reference/java/util/concurrent/BlockingQueue.html
    BlockingQueue  loadViewQueue;


    @Override
    public int getCount() {
        return countOfYourData;
    }

    @Override
    public View getView( int position, View convertView, ViewGroup parent ) {
        if( convertView == null ) {
            convertView = LayoutInflater.from(getContext()).inflace( yourresId, parent, false );
        }

        // if you don't use ViewHolder patter you may store position as shown below
        // otherwise create appropriate field in ViewHolder class and store position there
        convertView.setTag(position);
        loadViewQueue.put(convertView);
    }

    class LoadingAsyncTask extends AsyncTask< View, Object, Void > {
        Integer position;
        @Override
        protected Void doInBackground( View... params ) {

            while(true) {
                View view = loadViewQueue.take();
                position = ( Integer )view.getTag();
                Data data = loadDataFromDb(position);
                publishProgress( position, data );
            }
            return null;
        }

        @Override
        protected void onProgressUpdate( Object result ) {
            applyDataToListItem((Integer)result[0], (Data)result[1]);
        }
    }

конечно очередь должна быть синхронизирована и добавлена ​​проверка ошибок.

Другие вопросы по тегам