android - Android ListView 在向下滚动时显示不正确的内容,然后在向上滚动时显示正确的内容
问题描述
作为 Android Studio 的新手,我目前对相同代码的不同行为感到困惑。如果代码相同,向上或向下滚动时结果如何不同?这可能是一些 Android 视图回收问题吗?
从 Web 上的 JSON 文件解析数据后,自定义适配器会读取并设置 ListView 中每一行的相应数据。每行有四条信息:1)标题,2)描述,3)图像 URL(通过毕加索显示)和 4)分配给按钮的 setOnClickListener 的外部 URL,通过浏览器意图。
虽然每一行的标题、描述和图像都可以正常工作,但对于分配给 Button 的 setOnClickListener 的 URL,恰好有一个非常奇怪、奇怪且到目前为止(对我而言)莫名其妙的行为:
1) 首次呈现ListView时,在浏览器点击Button打开URL时,系统打开的不是该一项的正确URL,而是Array中NEXT项对应的URL。
2)如果向下滚动一点,然后再次向上滚动,当点击同一个按钮(之前导致错误的 URL)时,这一次它确实会打开正确的 URL。
在相同的代码中,在相同的运行中,Button 在向下滚动时会显示不正确的 URL,然后在向上滚动时会显示正确的 URL。
到目前为止,我的提示是考虑这可能与 Android 的 Recycling Views 功能有关,但我仍然不知道如何纠正这个问题。
我确实在 CustomAdapter.java 中包含了以下代码行,希望防止 Recycling:
@Override
public int getViewTypeCount() {
return getCount();
}
@Override
public int getItemViewType(int position) {
return position;
}
然而,包含或排除上述代码并没有实际的区别。
我的 CustomAdapter.java 的完整代码如下:
package ca.costari.apps.ge3000;
import android.annotation.SuppressLint;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.os.Build;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.TextView;
import com.squareup.picasso.Picasso;
import java.util.ArrayList;
import java.util.Objects;
public class CustomAdapter extends BaseAdapter {
private Context context;
private ArrayList<PlayersModel> playersModelArrayList;
private String clickurl;
public CustomAdapter(Context context, ArrayList<PlayersModel> playersModelArrayList) {
this.context = context;
this.playersModelArrayList = playersModelArrayList;
}
@Override
public int getViewTypeCount() {
return getCount();
}
@Override
public int getItemViewType(int position) {
return position;
}
@Override
public int getCount() {
return playersModelArrayList.size();
}
@Override
public Object getItem(int position) {
return playersModelArrayList.get(position);
}
@Override
public long getItemId(int position) {
return 0;
}
@SuppressLint("InflateParams")
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if (convertView == null) {
holder = new ViewHolder();
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
convertView = Objects.requireNonNull(inflater).inflate(R.layout.lv_item, null, true);
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
holder.procesa_titulo = Objects.requireNonNull(convertView).findViewById(R.id.id_titulo);
}
holder.procesa_mensajepromo = convertView.findViewById(R.id.id_mensajepromo);
holder.procesa_imagenurl = convertView.findViewById(R.id.imageViewPromo);
convertView.setTag(holder);
}else {
// the getTag returns the viewHolder object set as a tag to the view
holder = (ViewHolder)convertView.getTag();
}
clickurl = playersModelArrayList.get(position).getClickURL();
holder.procesa_clickurl = convertView.findViewById(R.id.id_clickurl);
holder.procesa_clickurl.setOnClickListener(new View.OnClickListener() {
public void onClick(View v) {
AsignaBotonUrl(v, clickurl);
}
});
holder.procesa_titulo.setText(playersModelArrayList.get(position).getTitulo());
holder.procesa_mensajepromo.setText(playersModelArrayList.get(position).getMensajePromo());
// Picasso to display image from URL
Picasso.with(holder.procesa_imagenurl.getContext())
.load(playersModelArrayList.get(position).getImageURL())
.placeholder(R.drawable.echale_un_ojo_te_interesa)
.into(holder.procesa_imagenurl);
return convertView;
}
private void AsignaBotonUrl(View view, String url) {
Intent browserIntent = new Intent(
Intent.ACTION_VIEW,
Uri.parse(url));
view.getContext().startActivity(browserIntent);
}
private class ViewHolder {
protected TextView procesa_titulo, procesa_mensajepromo;
protected ImageView procesa_imagenurl;
protected Button procesa_clickurl;
}
}
上面的代码将值分配给以下 ListView xml 中包含的项目:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="10dp"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="#fdfdfd"
android:orientation="vertical"
android:layout_marginBottom="30dp"
tools:ignore="UselessParent">
<TextView
android:id="@+id/id_titulo"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textColor="#2f2f2f"
android:layout_marginTop="10dp"
android:layout_marginBottom="20dp"
android:gravity="center_vertical"
android:textSize="20sp"
android:paddingLeft="10dp" />
<ImageView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/imageViewPromo"
android:layout_marginBottom="20dp"
android:contentDescription="@string/echale_un_ojo" />
<TextView
android:id="@+id/id_mensajepromo"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:textColor="#2f2f2f"
android:gravity="center_vertical"
android:textAppearance="?android:attr/textAppearanceMedium"
android:paddingLeft="10dp"
android:layout_marginBottom="10dp" />
<Button
android:id="@+id/id_clickurl"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:textSize="24sp"
android:text="@string/conocer_mas"
android:drawableRight="@drawable/open_web"
android:layout_marginBottom="30dp" />
</LinearLayout>
</LinearLayout>
这是我在 StackOverflow 上的第一篇文章,因此请您耐心等待,因为我可能没有正确地提出我的问题,如果我可能需要在此处包含更多代码或提供更多信息,请告诉我。
万分感谢!!!
解决方案
更改clickurl
为最终局部变量,因此删除
private String clickurl
并且只需使用它,getView
以便每个侦听器都有一个唯一的 URL。
final String clickurl = playersModelArrayList.get(position).getClickURL();
照原样,您的 onClickListener 使用clickurl
(保存在 Adapter 类中的那个)的单个实例,因此它总是会使用上次调用中设置的任何getView
.
推荐阅读
- vim - 如何为静态站点生成器 Hugo 运行更漂亮的降价文件?
- image-segmentation - 用于 UNET 的 VIA 注释工具 json 输出文件
- c++ - uncrustify:我可以将其配置为完全重新格式化一行吗?
- php - Quickbooks PHP API:刷新 OAuth 2 令牌失败
- r - 为数百万年的数据指定时间序列参数
- javascript - React.js 告诉你取消承诺。官方承诺不能取消。我应该怎么做?
- flutter - Flutter中打开键盘时如何让屏幕保持在固定位置?
- ruby-on-rails - 如何使用 Rails 6 安装 easydropdown
- laravel - '包 zendframework/zend-dictoros 被废弃,你应该避免使用它。改用 laminas/laminas-diactoros。
- c# - 如何检查 Redis 频道是否已使用 StackExchange.Redis 订阅?