5

このgetViewを備えたアダプターがあります:

public View getView(int position, View convertView, ViewGroup parent) {
    Log.d("getView gv", position+"");

    NewsLine holder = null;

    if (convertView == null) {
        LayoutInflater inflater = ((Activity) context).getLayoutInflater();
        convertView = inflater.inflate(R.layout.grid_entry, parent, false);
        holder = new NewsLine();

        holder.iv= (ImageView) convertView.findViewById(R.id.photo);

        convertView.setTag(holder);
    } else {
        holder = (NewsLine) convertView.getTag();
    }

    NewsItem item=news.ITEMS.get(position);
    //---------

    if(item.imgurl!=null && item.imgurl.compareToIgnoreCase("null")!=0) 
    {
        holder.iv.setVisibility(View.VISIBLE);     
        mMemoryCache.loadBitmap(item.imgurl, holder.iv,position);
    }
    else
        holder.iv.setVisibility(View.INVISIBLE);
    //-------------




    return convertView;
}

2 つの問題があります。

  1. getView は位置 0 に対して数回呼び出されます (ビットマップが LruCache に見つからない場合、AsyncTask でダウンロードされます)。その位置で数回再起動するアニメーション (0-1 のアルファ) があります。

  2. ビューをリサイクルしているため、古い imageView コンテンツが一瞬表示されることがあります。

//----

そして、これがキャッシュクラスです(ヒープのみ):

public class SetImgAT extends LruCache<String, Bitmap> {
private static SetImgAT instance;   
private Animation FadeInAnimation;

private SetImgAT(int size, Context context) {
    super(size);
    FadeInAnimation = AnimationUtils.loadAnimation(context, R.anim.fadein);
}

public static synchronized SetImgAT getInstance(int size, Context context) {
    if (instance == null) {
        instance = new SetImgAT(size, context);
    }
    return instance;
}

@Override
protected int sizeOf(String key, Bitmap value) {
    return (value.getRowBytes() * value.getHeight());
}

public void loadBitmap(String url, ImageView imageView,int pos) {
    Bitmap bitmap = instance.get(url.hashCode() + "");
    if (bitmap != null) {
        Log.d("ImageCache", "hit - "+url.hashCode()+"pos:"+pos);
        imageView.setImageBitmap(bitmap);

        imageView.invalidate();
    } else {
        Log.d("ImageCache", "miss");
        BitmapWorkerTask task = new BitmapWorkerTask(imageView);
        task.execute(url);
    }
}

class BitmapWorkerTask extends AsyncTask<String, Void, Bitmap> {
    ImageView mImageView;

    public BitmapWorkerTask(ImageView imageView) {
        mImageView = imageView;
    }

    @Override
    protected Bitmap doInBackground(String... url) {
        Bitmap Picture = null;


        if (url[0] != null && url[0].compareToIgnoreCase("null") != 0) {
            Log.d("GetBMP from", url[0]);

            URL img_value = null;
            try {
                img_value = new URL(url[0]);
            } catch (MalformedURLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            try {
                Picture = BitmapFactory.decodeStream(img_value
                        .openConnection().getInputStream());
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            if (Picture == null) {
                Log.d("deb", "no bitmap");
            } else {
                Log.d("got deb", "got bitmap to "+url[0].hashCode());                   
                instance.put(url[0].hashCode()+"", Picture);
            }

        }
        return Picture;
    }

    @Override
    protected void onPostExecute(Bitmap result) {
        // super.onPostExecute(result);
        if (result != null) {
            Log.d("deb", "set bitmap");
            mImageView.setImageBitmap(result);
            //mImageView.startAnimation(FadeInAnimation);
        }

    }
}

//----------------

}

ありがとうございました!:)

4

1 に答える 1

0

前後にスクロールしたり、無計画に notifyDataSetChanged() を呼び出したりすると、同様の動作が見られました。

フェードインアニメーションに加えて、このケースを非常にうまく処理するため、代わりに Picasso を使用することをお勧めします。

getView() のワンライナー:

Picasso.with(context).load(urlToLoad).into(imageView);

参照: http://square.github.io/picasso/

于 2014-04-28T21:23:38.713 に答える