Android-在自定义适配器中设置文本时使用NPE

Android-在自定义适配器中设置文本时使用NPE,android,nullpointerexception,custom-adapter,Android,Nullpointerexception,Custom Adapter,这在堆栈溢出问题上已经被问过很多次了,我已经尝试过很多次了,但是当我在自定义适配器中调用TextView上的setText方法时,仍然会出现空指针异常。我必须显示所有收件箱信息,第一个文本视图显示发件人的号码,第二个文本视图显示短信。这是我的适配器: public class MyArrayAdapter extends ArrayAdapter<Sms> { private final Context context; private fin

这在堆栈溢出问题上已经被问过很多次了,我已经尝试过很多次了,但是当我在自定义适配器中调用TextView上的setText方法时,仍然会出现空指针异常。我必须显示所有收件箱信息,第一个文本视图显示发件人的号码,第二个文本视图显示短信。这是我的适配器:

public class MyArrayAdapter extends ArrayAdapter<Sms> {
          private final Context context;
          private final List<Sms> msgs;
          private LayoutInflater inflater;

          public MyArrayAdapter(Context context, List<Sms> msgs) {
            super(context, R.layout.list_item, msgs);
            this.context = context;
            this.msgs = msgs;
            inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
          }



          @Override
          public View getView(int position, View convertView, ViewGroup parent){
              Sms s = getItem(position);
              View vi = convertView;
              ViewHolder vh = new ViewHolder();
              if (vi == null) {
                  vi = getLayoutInflater().inflate(R.layout.list_item, parent, false);

                  vh.number = (TextView) findViewById(R.id.number);
                  vh.msg = (TextView) findViewById(R.id.sms);
                  vi.setTag(vh);
              }
              try{
                  vh.number.setText(s.getReadState());
                  vh.msg.setText(s.getMsg());
              }
              catch (NullPointerException e){
                  e.printStackTrace();
              }

              return vi;

          }


    }

    public static class ViewHolder{
        TextView number;
        TextView msg;
    }
第79行这是一个:

vh.number.setText(s.getReadState());

引发异常的原因是
ViewHolder.message
ViewHolder.msg
vi!=空
。根本原因是您没有正确应用ViewHolder图案:

您不应该在每次迭代中创建一个新的
ViewHolder
实例,这会破坏模式的目的。相反,如果传入了回收视图,则从先前标记的视图中检索
ViewHolder
。此
else
案例目前在您的代码中完全缺失,而try-catch无法神奇地使您的代码正常工作

如果您进行了上面概述的更改,那么您应该得到类似于以下内容的结果。请注意,我是直接在浏览器中输入的,因此请注意任何明显的打字错误

@Override
public View getView(int position, View convertView, ViewGroup parent){
    Sms s = getItem(position);
    View vi = convertView;
    ViewHolder vh;
    if (vi == null) {
        vi = getLayoutInflater().inflate(R.layout.list_item, parent, false);
        vh = new ViewHolder();            
        vh.number = (TextView) vi.findViewById(R.id.number);
        vh.msg = (TextView) vi.findViewById(R.id.sms);
        vi.setTag(vh);
    } else {
        vh = (ViewHolder) vi.getTag(); 
    }

    vh.number.setText(s.getReadState());
    vh.msg.setText(s.getMsg());

    return vi;

}

请发布日志猫。@Prmths添加了日志猫。请看一看。我编辑代码的方式与您提到的相同,但我仍然得到相同的异常和相同的日志。try catch只是为了避免在logcat而不是电话上看到那个丑陋的错误。
R.layout.list_item
的内容是什么?理论上,如果
number==null
,同一行仍可能抛出NPE,如果在布局中找不到视图,则可能发生这种情况。不管怎么说,这只是一个暗箱操作-请在附加了调试器的情况下逐步检查您的代码,然后发回
null
。我通过调试器运行了我的代码,看起来即使在为两个文本视图设置findViewById之后,它们也将变为null。ie debugger显示vh.msg=null和vh.number=nullAh,我知道可能出了什么问题:刚刚膨胀的布局可能会发生
findviewbyd()
调用。现在,查找在您的
活动中进行。将
findViewById(R.id.number)
更改为
vi.findViewById(R.id.number)
,并对
sms执行相同操作。我还更新了我的答案,以反映这一更正。哦,我的上帝!我会为此发疯的。这很有效。非常感谢D(由于声誉低,无法升级:P,但标记为已接受)
@Override
public View getView(int position, View convertView, ViewGroup parent){
    Sms s = getItem(position);
    View vi = convertView;
    ViewHolder vh;
    if (vi == null) {
        vi = getLayoutInflater().inflate(R.layout.list_item, parent, false);
        vh = new ViewHolder();            
        vh.number = (TextView) vi.findViewById(R.id.number);
        vh.msg = (TextView) vi.findViewById(R.id.sms);
        vi.setTag(vh);
    } else {
        vh = (ViewHolder) vi.getTag(); 
    }

    vh.number.setText(s.getReadState());
    vh.msg.setText(s.getMsg());

    return vi;

}