未调用自定义适配器 getView() 方法

Posted

技术标签:

【中文标题】未调用自定义适配器 getView() 方法【英文标题】:Custom Adapter getView() method is not called 【发布时间】:2013-04-26 15:20:41 【问题描述】:

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

没有错误,但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;
    


【问题讨论】:

【参考方案1】:

getView 未被调用的唯一原因是:

    getCount 返回 0。 您忘记在ListView 上拨打setAdapter。 如果ListView 的可见性(或其容器的可见性)为GONE。感谢@TaynãBonaldo 提供的宝贵意见。 ListView 未附加到任何视口布局。也就是说,mListView = new ListView(...) 不使用myLayout.addView(mListView)

onPostExcute 中,创建CarListAdapter 的新实例后,我建议您将新实例更新为ListView。确实需要再打电话

 mList.setAdapter(adapter);

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

编辑2:

这同样适用于RecyclerView。确保

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

【讨论】:

它起作用了,但你能解释一下为什么,因为在片段生命周期中 onCreate() 是在 onCreateView() 之前调用的,我在 onCreate() 中调用 asyncTask 并在 onCreateView() 中设置适配器,然后调用它。 @Ravi 首先,没有人可以向您保证 AsyncTask 执行在调用 onCreateView 之前或之后完成。这就是异步的意思。二,在 onPostExcute 你创建一个新的 CarListAdapter 并且你的 ListView 必须被告知适配器已经改变 我的意思是通过在 onPosExecute 方法中设置适配器是解决这种情况的好方法,或者有更好的架构来避免此类问题 你能改写最后的评论吗? 我面临的另一个问题。如果您使用约束布局,那么您必须为 listView 提供约束,或者您必须为 listView 提供宽度和高度。宽度和高度是必需的,但如果你不给 A.Studio 只会给出警告。【参考方案2】:

您必须验证列表是否包含元素,在将项目添加到列表时可能会出错。 要验证,请使用以下方法:

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);
    
);

希望这对遇到类似问题的其他人有所帮助

【讨论】:

我看不出这有什么帮助。 ASAIK,视图是在 Activity 布局膨胀时创建的。我认为没有必要进一步“等待”它的创建。我面临着类似的问题,这不是解决方案。【参考方案4】:

您在构造函数中缺少超类。请参阅下面的示例:

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

【讨论】:

我要补充一点:adapter = new CarListAdapter... 实际上并没有为视图设置适配器。所以他在 onCreateView 和 onPostExecute 之间存在竞争条件 你说得对,我没有发现他在执行后没有将新适配器绑定到列表视图。黑带ftw @Raanan 你能详细说明应该做些什么来避免这种竞争条件并解释..我是android开发的新手..如果你能解释一下,那就是gr8..谢谢 setAdapter 将适配器绑定到列表视图(告诉列表视图它显示的数据来自何处)。在 onCreate() 中,您正在启动 aSyncTask 以下载数据,然后在 onCreateView 中设置适配器。但是,如果创建适配器(在 aSyncTask 中)比到达 onCreateView() 函数花费的时间更长,则列表视图将在填充汽车数据之前设置适配器,即它是空的。简单的解决方案是仅在您知道已成功创建适配器之后才绑定适配器。即正如黑带所说,只需将适配器绑定在 onPostExecute() :) 另一种选择是在 onCreate 中创建空适配器并将其绑定到 onCreateView 中,在 onPostExecute 中使用接收到的数据更新现有适配器并调用 notifyDataSetChanged,这将使适配器使用新信息刷新列表。如果您以这种方式实现它,您还可以稍后根据需要刷新数据,而无需创建新的适配器。【参考方案5】:

你一直在做的是

在您的适配器中

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

【讨论】:

【参考方案6】:

我遇到了同样的问题。在尝试了上面的所有提示之后,我的 getView 仍然没有被调用。所以我试图删除我在 ListView 之外使用的 ScrollView。然后 getView 运行良好。只是为了增加一种可能性。我希望能帮助别人。

【讨论】:

以上是关于未调用自定义适配器 getView() 方法的主要内容,如果未能解决你的问题,请参考以下文章

listview 项目未在自定义适配器的 getview 方法中显示分配的值

多次调用自定义列表视图适配器 getView 方法,并且顺序不一致

自定义列表视图适配器 getView 方法被多次调用,并且顺序不一致

在 notifyDataSetChanged 上未调用 GetView,ListView 卡住

创建您自己的自定义适配器时,getView() 方法如何工作?

NullPointerException 自定义列表视图适配器