c# - Android,RecyclerView:正在加载正确的视图,但在错误的位置充气
问题描述
我正在制作一个可以在您的手机上播放音乐文件的应用程序。它在回收站视图中列出所有歌曲及其艺术家姓名、歌曲名称和专辑封面(如果有)。
在我尝试加载专辑封面之前,一切正常。虽然一开始(在我滚动之前)一切看起来都很好,但经过一些滚动后,封面开始出现在它们不应该出现的地方。所以说我刚刚看过歌曲 X 的专辑封面,然后在滚动一些之后,歌曲 X 的同一封面出现在歌曲 y 旁边。这特别奇怪,因为它确实正确加载了专辑名称和其他所有内容。它只是混淆位图。这就是我初始化回收站视图的方式:
private async Task InitRecView()
{
await Task.Run(() =>
{
// Instantiate the adapter and pass in its data source:
LinearLayout lnBg = (LinearLayout)FindViewById(Resource.Id.background_recView);
mAdapter = new PhotoAlbumAdapter(GetSortedListWithAllSongs(), this, dbSeekObj, seekObj, mAudioManager, this, lnBg);
// Get our RecyclerView layout:
mRecyclerView = FindViewById<RecyclerView>(Resource.Id.recyclerView);
// Plug the adapter into the RecyclerView:
mRecyclerView.SetAdapter(mAdapter);
mRecyclerView.SetItemViewCacheSize(15);
mRecyclerView.DrawingCacheEnabled = true;
mRecyclerView.DrawingCacheQuality = DrawingCacheQuality.High;
mLayoutManager = new LinearLayoutManager(this);
mRecyclerView.SetLayoutManager(mLayoutManager);
});
}
这是我的适配器,我在其中检索数据(问题可能出现在“SetContentAsync()”中
public override RecyclerView.ViewHolder OnCreateViewHolder(ViewGroup parent, int viewType)
{
View itemView = LayoutInflater.From(parent.Context).
Inflate(Resource.Layout.CardView, parent, false);
PhotoViewHolder vh = new PhotoViewHolder(itemView, mp3Obj, act, reader, db, seekObj, audioManager, lnBg, ctx);
return vh;
}
public override void OnBindViewHolder(RecyclerView.ViewHolder holder, int position)
{
PhotoViewHolder vh = holder as PhotoViewHolder;
SetContent(vh, position);
}
private async void SetContent(PhotoViewHolder vh, int position)
{
await SetContentAsync(vh, position);
}
private async Task SetContentAsync(PhotoViewHolder vh, int position)
{
string SongName = "";
string ArtistName = "";
Bitmap bitmap = null;
byte[] data = null;
try
{
reader.SetDataSource(mp3Obj[position].Mp3Uri);
}
catch { }
await Task.Run(() => // cause problems with the reload
{
SongName = reader.ExtractMetadata(MediaMetadataRetriever.MetadataKeyTitle);
ArtistName = reader.ExtractMetadata(MediaMetadataRetriever.MetadataKeyArtist);
data = reader.GetEmbeddedPicture();
if (data != null)
{
bitmap = BitmapFactory.DecodeByteArray(data, 0, data.Length);
}
});
((Activity)ctx).RunOnUiThread(() =>
{
vh.SongName.SetTypeface(tf, TypefaceStyle.Normal);
vh.AristName.SetTypeface(tf, TypefaceStyle.Normal);
vh.SongName.Text = SongName;
vh.AristName.Text = ArtistName;
if (bitmap != null)
{
ConvertBitmapToBackground(bitmap, vh); // Set As Backgorund, blurry and black ( just sets the variable)
CutImageIntoForm(bitmap, vh); // Set as the musical note button
}
});
}
这是 viewholder,除了初始化视图之外什么都没有发生:
public PhotoViewHolder(View itemView, List<MP3objectSmall> mp3Obj, Activity act, MediaMetadataRetriever reader, DataBase db, List<SeekObj> seekObj, AudioManager audioManager, LinearLayout lnBg, Context ctx) : base(itemView)
{
// Locate and cache view references:
SongName = itemView.FindViewById<TextView>(Resource.Id.textView);
AristName = itemView.FindViewById<TextView>(Resource.Id.textView2);
lnContainer = itemView.FindViewById<LinearLayout>(Resource.Id.linlay_cardview);
CoverArt = itemView.FindViewById<ImageButton>(Resource.Id.musical_note);
this.mp3Obj = mp3Obj;
this.act = act;
this.reader = reader;
this.db = db;
this.seekObj = seekObj;
this.ctx = ctx;
this.audioManager = audioManager;
this.lnBg = lnBg;
lnContainer.Click += delegate
{
int pos = AdapterPosition;
ClickEvent(pos, AristName.Text, SongName.Text, CoverArt, lnBg);
};
}
如果有人可以在这里帮助我。为什么我的回收站视图会混淆位图并将它们放入错误的布局和/或将它们加倍?
谢谢!
编辑:
我注意到相同项目的再次出现总是在相同的时间间隔内发生。对我来说,每 24 个项目包含该项目之前第 24 个项目位置的图像。我还读到这是因为回收者视图是回收者所说的视图,但我不明白的是如何找出一个项目是否被回收,何时回收,只需使用新数据......
解决方案
由于您使用异步方法,因此难怪它会显示旧数据。在viewHolder中创建clear
方法并在onViewRecycledMethod中调用:
public class TextRecyclerAdapter extends RecyclerView.Adapter<TextRecyclerAdapter.TextViewHolder> {
...
@Override
public void onViewRecycled(@NonNull TextViewHolder holder) {
//clear your view here
holder.clear();
}
....
public static class TextViewHolder extends RecyclerView.ViewHolder {
...
public void clear(){
//clear your view here
_textView.setText("");
}
private TextView _textView;
...
}
更新 这也可能发生,因为来自异步调用的旧数据比新数据来得晚。您需要使用取消令牌取消回收的旧请求。
推荐阅读
- laravel - Laravel:附加属性会导致 n+1 问题吗?
- fortran - Fortran 中的表格输出,对意外值具有鲁棒性?
- css - 为什么 CSS 尺寸不适用于低宽度?
- git - 如何通过远程设置 git committer 电子邮件
- flutter - 推新屏后如何重建状态
- c# - 如何使用具有权限的电子邮件办公室互操作访问正文属性
- php - 如果有字母字符,MySQL Select 语句不起作用
- css - 将 CSSgram Instagram Willow 过滤器转换为 SVG 过滤器
- hard-coding - 究竟什么是硬编码?
- excel - 需要帮助为确定的条件过滤 Excel 工作表