自定义适配器的getView()方法没有被调用

42

这是我设置自定义适配器到列表的片段代码。

没有错误,但是ListView为空。我已经实现了getCount()方法,并且返回了我的ArrayList中正确的项目数量。在Logcat中我看不到("Inside", "GetView")

片段

public class ServiceCarListFragment extends Fragment {

    private String url;
    private ArrayList<CarDetail> carDetailList = new ArrayList<CarDetail>();
    private CarListAdapter adapter;
    private ListView mList;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        url = getActivity().getIntent().getStringExtra("url");
        new DownloadCarDetail().execute(url);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        View v = inflater.inflate(R.layout.fragment_service_car_list, container, false);
        mList = (ListView) v.findViewById(R.id.list);
        mList.setAdapter(adapter);

        for (CarDetail car : carDetailList) {
            // START LOADING IMAGES FOR EACH STUDENT
            car.loadImage(adapter);
        }
        return v;
    }

    class DownloadCarDetail extends AsyncTask<String, String, ArrayList<CarDetail>> {

        @Override
        protected ArrayList<CarDetail> doInBackground(String... params) {
            // TODO Auto-generated method stub
            ArrayList<CarDetail> carDetailList = JsonParser.parseJson(params[0]);
            return carDetailList;
        }

        @Override
        protected void onPostExecute(ArrayList<CarDetail> carDetailList) {
            // TODO Auto-generated method stub
            ServiceCarListFragment.this.carDetailList = carDetailList;
            Log.d("dccs", String.valueOf(ServiceCarListFragment.this.carDetailList.size()));
            adapter = new CarListAdapter(getActivity(), ServiceCarListFragment.this.carDetailList);
            Log.d("dccs", String.valueOf((adapter.getCount())));
        }

    }
}

自定义适配器

public class CarListAdapter extends BaseAdapter {

    private ArrayList<CarDetail> items = new ArrayList<CarDetail>();
    private Context context;

    public CarListAdapter(Context context, ArrayList<CarDetail> items) {
        this.context = context;
        this.items = items;
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return items.size();
    }

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return items.get(position);
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // TODO Auto-generated method stub
        Log.d("Inside", "GetView");
        LayoutInflater mInflater = (LayoutInflater) context.getSystemService(Activity.LAYOUT_INFLATER_SERVICE);
        ViewHolder holder = null;
        CarDetail car = items.get(position);

        if (convertView == null) {
            convertView = mInflater.inflate(R.layout.car_list_row, null);
            holder = new ViewHolder();
            holder.tvCarName = (TextView) convertView.findViewById(R.id.tvCarName);
            holder.tvDailyPriceValue = (TextView) convertView.findViewById(R.id.tvWeeklyPriceValue);
            holder.tvWeeklyPriceValue = (TextView) convertView.findViewById(R.id.tvWeeklyPriceValue);
            holder.imgCar = (ImageView) convertView.findViewById(R.id.imgCar);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        holder.tvCarName.setText(car.getCarName());
        if (car.getImage() != null) {
            holder.imgCar.setImageBitmap(car.getImage());
        } else {
            // MY DEFAULT IMAGE
            holder.imgCar.setImageResource(R.drawable.ic_action_call);
        }

        return convertView;
    }

    static class ViewHolder {
        TextView tvCarName;
        TextView tvDailyPriceValue;
        TextView tvWeeklyPriceValue;
        ImageView imgCar;
    }

}
6个回答

169
getView方法未被调用的唯一原因有:
  1. getCount返回0。
  2. 你忘记在ListView上调用setAdapter
  3. 如果ListView(或其容器)的可见性是GONE。感谢@TaynãBonaldo提供的宝贵意见。
  4. ListView未附加到任何视口布局中。也就是说,使用mListView = new ListView(...) 没有使用myLayout.addView(mListView)

onPostExecute中,在创建CarListAdapter的新实例后,建议您将新实例更新到您的ListView中。确实,您需要再次调用:

 mList.setAdapter(adapter);

编辑:setAdapter应始终在UI线程上调用,以避免意外行为

编辑2:

对于RecyclerView也适用。请确保:

  • getItemCount返回大于0的值(通常是数据集大小)
  • setLayoutManagersetAdapter都必须在UI线程上调用
  • 小部件的可见性必须设置为VISIBLE

1
@Ravi 首先,没有人能保证在调用onCreateView之前或之后AsyncTask的执行已经完成。这就是异步的含义。其次,在onPostExecute中,您创建了一个新的CarListAdapter,您的ListView必须被告知适配器已更改。 - Blackbelt
我指的是在onPostExecute方法中设置适配器是解决这种情况的好方法,还是有更好的架构可以避免这些问题。 - Ravi
也许你的 listView 是空的。 - user3717188
这是解决这个问题最系统的方法。我逐一检查发现,在API调用的onPostExecute中我的视图被设置为gone。 - Shubham Chaudhary
1
我面临的另一个问题是:如果你使用了约束布局,那么你必须给listView设置约束,或者给它指定宽度和高度。虽然宽度和高度是必需的,但如果你没有指定,A.Studio 只会提示警告。 - ahmet
显示剩余6条评论

5

您必须验证列表是否有元素,以防在向列表添加项目时出现错误。 要进行验证,请使用以下方法:

adapter.getCount();

3

我曾经遇到过类似的问题。这里有一个简单的解决方法:

在您的onCreateView中,您需要等待视图被创建。所以请将以下几行代码进行更改:

mList = (ListView)v.findViewById(R.id.list);
mList.setAdapter(adapter);

请将上述两行代码修改为:

mList = (ListView)v.findViewById(R.id.list);
mList.post(new Runnable() {
    public void run() {
        mList.setAdapter(adapter);
    }
});

希望这能帮助到遇到类似问题的其他人。

我不明白这有什么帮助。据我所知,视图是在Activity布局被填充时创建的。我不认为还需要进一步“等待”它的创建。我遇到了类似的问题,这并不是一个解决方案。 - Bamaco

0

你在构造函数中缺少了超类。请参考下面的示例:

public AppDataAdapter(Activity a, int textViewResourceId, ArrayList<AppData> entries) {
    super(a, textViewResourceId, entries);
    this.entries = entries;
    this.activity = a;
}

我想补充一点: adapter = new CarListAdapter... 实际上并没有为视图设置适配器。因此,在onCreateView和onPostExecute之间存在竞争条件。 - Raanan
你说得对,我没有注意到他在post execute中没有将新的适配器绑定到listview。黑带胜利。 - rcbevans
@Raanan,您能详细说明如何避免这种竞态条件并解释一下吗?我是Android开发的新手。如果您能解释一下,那就太好了。谢谢。 - Ravi
1
setAdapter将适配器绑定到ListView(告诉ListView它正在显示的数据来自哪里)。在onCreate()中,您正在启动aSyncTask以下载数据,然后在onCreateView中设置适配器。但是,如果创建适配器(在aSyncTask中)所需的时间比达到onCreateView()函数所需的时间长,则ListView将在其填充汽车数据之前设置适配器,即为空。简单的解决方案是仅在成功创建适配器后才绑定适配器。即如blackbelt所说,在onPostExecute()中绑定适配器 :) - rcbevans
2
另一个选项是在onCreate中创建空的Adapter,在onCreateView中进行绑定,在onPostExecute中使用接收到的数据更新现有的Adapter,并调用notifyDataSetChanged,这将使适配器使用新信息刷新列表。如果您以这种方式实现,您还可以稍后刷新数据而无需创建新的适配器。 - Raanan
显示剩余2条评论

0
你所做的是:
在你的适配器中。
public CarListAdapter(Context context , ArrayList<CarDetail> items) {

    this.context = context;
    this.items = items;

}

在你的碎片中

adapter = new CarListAdapter(getActivity(),ServiceCarListFragment.this.carDetailList);

我希望你会使用FragmentActivity

你需要调用

adapter = new CarListAdapter(YOUR_ACTIVITY_CONTEXT, carDetailList);

其中YOUR_ACTIVITY_CONTEXT将是您的FragmentActivity


0

我曾经遇到过同样的问题。尝试了以上所有提示后,我的getView仍然没有被调用。所以我尝试移除了我在ListView外部使用的ScrollView。然后getView就正常工作了。只是为了增加一种可能性。希望能帮助到某些人。


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