在Android网格视图中使用Universal Image Loader滚动时重复显示图像

8
我正在尝试使用 Universal Image Loader 在网格视图上加载来自互联网的图片,此处代码如下:

public View getView(int position, View converView, ViewGroup parent) {
        Log.v("Description", "Description is " + position);
        ViewHolder mVHolder;
        if (converView == null) {
            LayoutInflater vi = (LayoutInflater) conted.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            converView = vi.inflate(R.layout.customgrid, null);
            mVHolder = new ViewHolder();
            mVHolder.mImageView = (ImageView) converView
                    .findViewById(R.id.imgview);
            mVHolder.mTextView1 = (TextView) converView
                    .findViewById(R.id.textView1);
            mVHolder.mTextView2 = (TextView) converView
                    .findViewById(R.id.textView2);
            mVHolder.mTextView3 = (TextView) converView
                    .findViewById(R.id.textView3);

            converView.setTag(mVHolder);
        } else {
            mVHolder = (ViewHolder) converView.getTag();
        }
        // mVHolder.mImageView.setImageResource(mThumbIds[position]);
        // mVHolder.mImageView.setImageDrawable(LoadImageFromURL(
        // mThumbIds[position]));
        imageLoader.displayImage(mThumbIds[position],mVHolder.mImageView,options, animationListener);
        Log.v("Names",NAMES[position] + STATES[position] + CONSTITUENCY[position]);
        mVHolder.mTextView1.setText(NAMES[position]);
        mVHolder.mTextView2.setText(STATES[position]);
        mVHolder.mTextView3.setText(CONSTITUENCY[position]);
        return converView;
    }

然而,当我在列表中向下滚动时,第一行的第一张图片似乎会在大多数网格中重复出现。它并不总是保持这样 - 一段时间后它会恢复到原始图片。我的问题是,这是GridView视图膨胀或Universal Image Loader图像加载的问题吗?

这里的imageLoader是什么? - km86
@user3431672,imageloader是通用图像加载器 https://github.com/nostra13/Android-Universal-Image-Loader - Saty
看起来似乎与适配器有关。为了测试,请尝试删除视图的回收并查看图像是否正确显示。这可能会很慢,滚动可能会很卡顿。但它应该能够给我们一个问题出现的想法。然后我们就可以进行调试了。 - km86
2个回答

14
我认为您的问题在于未设置DisplayImageOptionsresetViewBeforeLoading()方法,如下所示:
     DisplayImageOptions options = new DisplayImageOptions.Builder()
    .showStubImage(R.drawable.stub_image)
    .showImageForEmptyUri(R.drawable.image_for_empty_url)
 ---->   .resetViewBeforeLoading(true)<----very important for recycling views
    .cacheInMemory()
    .cacheOnDisc()
    .imageScaleType(ImageScaleType.IN_SAMPLE_POWER_OF_2) // default
    .bitmapConfig(Bitmap.Config.ARGB_8888) // default
    .delayBeforeLoading(1000)
    .displayer(new SimpleBitmapDisplayer()) // default
    .build();

然后应用这个 DisplayImageOptions。

        imageLoader.displayImage(mThumbIds[position],mVHolder.mImageView,options,                   animationListener);

1
这并不完全是这种情况,但它与此有关。我已经删除了showStubImage(R.drawable.stub_image)以查看效果,然后忘记了...现在已经添加了,它不再重复显示图像,谢谢您。 - Saty
1
对我也解决了!然而上述方法已经过时。在 Git Hub 页面上有更新的版本可用。 - Skynet
1
嗨,我有类似的问题。我尝试了这个解决方案,但问题仍未解决。 - Vijay Vankhede
1
在处理大型图像时,使用 resetViewBeforeLoading 是一个不好的选择。 - Gilberto Ibarra

2

另一种解决方案是使用setTag()来标识视图:

class ViewHolder {
    ImageView thumbnail;
    public boolean isSameView(String tag) {
        if (tag == null) {
            return thumbnail.getTag() == null;
        }
        return tag.equals(thumbnail.getTag());
    }
}


protected void imageLoader(ViewHolder holder, String imageUrl) {
    if (!holder.isSameView(imageUrl)) {  // If image is loaded correctly skip else reload and set new image
        holder.thumbnail.setTag(imageUrl);
        imageLoader.displayImage(imageUrl, holder.thumbnail, new SimpleImageLoadingListener() {@
            Override
            public void onLoadingFailed(String imageUri, View view, FailReason failReason) {
                view.setTag(failReason.getType().name()); // You can handle errors and avoid repeating requests that return error codes
            }
        });
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接