9

我有一个奇怪的问题让我发疯。在我的 android 应用程序中,我定制了我自己的从 ArrayAdapter 扩展的适配器。我添加适配器的 ListView 的项目可以是标记文本(不可编辑)、可编辑文本或微调器。疯狂的是:当我滚动 ListView 时,有两个问题:

(1)微调器项目中显示的(选定的)值有时会发生变化,尽管我只滚动了!!当我单击微调器时,仍然显示旧的选定值(应该由微调器显示的那个)(2)当我滚动时,ListViewItems的顺序会发生变化!

=>但是适配器中的数据没有改变(数据本身和顺序都没有) - 所以它一定是视图本身的问题?!也许android在后台缓存并且没有尽快刷新ListViewItems或类似的东西?!

有人可以帮我吗?

多谢!

好的,我找到了一个不是很好的解决方案,但它确实有效。我只是不再使用 convertView 了,尽管这在内存和性能方面不是最理想的。在我的情况下应该没问题,因为我的 ListView 的最大项目数量是 15。这是我的适配器类:

 public class FinAdapter extends ArrayAdapter<Param>{
    public Param[] params;
    private boolean merkzettel;
    protected EditText pv;

    public FinAdapter(Context context, int textViewResourceId, Param[] items, boolean merkzettel) {
        super(context, textViewResourceId, items);
        this.params = items;
        this.merkzettel = merkzettel;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        final Param p = params[position];
        if(p != null){
            if(merkzettel){
                if (convertView == null) {
                    LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                    convertView = vi.inflate(R.layout.merk_det_item, null);
                }
                TextView tvl = (TextView) convertView.findViewById(R.id.paramM);
                TextView edl = (TextView) convertView.findViewById(R.id.param_valueM);
                TextView pal = (TextView) convertView.findViewById(R.id.param_unitM);
                if (tvl != null) {
                    tvl.setText(p.getName());                            
                }
                if(pal != null){
                    pal.setText(p.getUnit());
                }
                if(edl != null){
                    edl.setText(p.getDefData());
                }
            }
            else{
                if(p.isSelect()){

                if (convertView == null) {
                    LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                    convertView = vi.inflate(R.layout.fin_prod_list_item_select, null);
                }            
                TextView tvs = (TextView) convertView.findViewById(R.id.paramS);
                Spinner sp = (Spinner) convertView.findViewById(R.id.spinner_kalk);
                TextView paU = (TextView) convertView.findViewById(R.id.param_unitS);


                if (tvs != null) {
                    tvs.setText(p.getName());                            
                }
                if(paU != null){
                    paU.setText(p.getUnit());
                }
                if(sp != null){
                    String[] values = new String[p.getData().size()];
                    for(int i=0; i<values.length; i++){
                        values[i] = p.getData().get(i);
                    }
                    ArrayAdapter<String> adapter = new ArrayAdapter<String>(this.getContext(), android.R.layout.simple_spinner_item, values);
                    adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
                    sp.setAdapter(adapter);
                    sp.setSelection(p.getData().indexOf(p.getDefData()));
                    sp.setOnItemSelectedListener(new OnItemSelectedListener(){
                        public void onItemSelected(AdapterView<?> parent,
                                View convertView, int pos, long id) {
                            p.setDefData(p.getData().get(pos));
                            p.setChanged(true);
                        }
                        public void onNothingSelected(AdapterView<?> arg0) {
                            // TODO Auto-generated method stub

                        }

                    });
                }
            }       
            else if(p.isEdit()){
                if (convertView == null) {
                    LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                    convertView = vi.inflate(R.layout.fin_prod_list_item_edit, null);
                }
                TextView pa = (TextView) convertView.findViewById(R.id.param);
                pv = (EditText) convertView.findViewById(R.id.param_value);
                TextView paE = (TextView) convertView.findViewById(R.id.param_unit);
                if (pa != null) {
                    pa.setText(p.getName());                            
                }
                if(paE != null){
                    paE.setText(p.getUnit());
                }
                if(pv != null){
                    pv.setText(p.getDefData());
                    pv.setOnEditorActionListener(new OnEditorActionListener(){
                        public boolean onEditorAction(TextView convertView, int actionId,
                                KeyEvent event) {
                            // TODO Auto-generated method stub
                            p.setDefData(pv.getText().toString());
                            p.setChanged(true);
                            return false;
                        }                   
                    }); 
                }

            }
            else if(p.isLabel()){
                if (convertView == null) {
                    LayoutInflater vi = (LayoutInflater)getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                    convertView = vi.inflate(R.layout.fin_prod_list_item_label, null);
                }
                TextView tvl = (TextView) convertView.findViewById(R.id.paramL);
                TextView edl = (TextView) convertView.findViewById(R.id.param_valueL);
                TextView pal = (TextView) convertView.findViewById(R.id.param_unitL);
                if (tvl != null) {
                    tvl.setText(p.getName());                            
                }
                if(pal != null){
                    pal.setText(p.getUnit());
                }
                if(edl != null){
                    edl.setText(p.getDefData());
                }   
            }}
        }
        return convertView;
    }
}
4

3 回答 3

5

我对列表中的多个项目类型有类似的问题。在我的情况下,列表项是一个部分(标签)项或一个常见的列表项。

要使用此类列表,您应该重写getViewTypeCountgetItemViewType方法。像这样的东西:

private static final int ITEM_VIEW_TYPE_ITEM = 0;
private static final int ITEM_VIEW_TYPE_SEPARATOR = 1;

@Override
public int getViewTypeCount() {
    return 2;
}

@Override
public int getItemViewType(int position) {
    return this.getItem(position).isSection() ? ITEM_VIEW_TYPE_SEPARATOR : ITEM_VIEW_TYPE_ITEM;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    final Item item = this.getItem(position);

    if (convertView == null) {
        convertView = mInflater.inflate(item.isSection() ? R.view1 : R.view2, null);
    }

    if(item.isSection()){
        //...
    }
    else{
        //...
    }

    return convertView;
}

然后 convertView 参数将始终正确并包含您需要的类型。

还有一件事:public Param[] params当您已经在基类中拥有该字段时,您显式添加了该字段ArrayAdapter<Param>

我建议从BaseAdapter类继承。

编辑: 这是您可以尝试使用的代码来完成您的Spinner工作:

sp.setTag(p);
sp.setOnItemSelectedListener(new OnItemSelectedListener(){
public void onItemSelected(AdapterView<?> parent, View convertView, int pos, long id) {
    Param currentItem = (Param)parent.getTag();
    currentItem.setDefData(currentItem.getData().get(pos));
    currentItem.setChanged(true);
}
//...

尝试使用getTagsetTag方法的组合。我记得我在事件处理程序和最终变量方面也有类似的问题,但我完全忘记了它们的原因,所以我无法准确解释为什么会发生这种情况。

于 2012-08-16T08:33:58.523 回答
1

正如 Sam 所说,Android 系统总是尽可能地尝试回收以节省资源。这意味着,开发人员需要自己跟踪ViewListView对于实体,或所谓的表示数据结构,您可以有一些东西来跟踪选择/取消选择状态,例如:

class PresentationData {
    // other stuffs..
    boolean isSelected = false;
}

将这些数据结构映射到Adapter,如果单击了任何项目,则将状态设置为 true:listPresentationData.get(selected_position).isSelected = true

好的,所以在getView()适配器中,为您的数据正确跟踪演示文稿。

if(listPresentationData.get(position).isSelected) 
{ 
    // set background color of the row layout..or something else 
}

此外,值得一提的是,最好为每个视图都有一个内存缓存,通常也称为ViewHolder提高性能。

于 2012-08-15T16:06:07.967 回答
0

正如大家所说,Android 回收视图。这意味着什么?想象一下,您的列表中有 10 个项目,但屏幕上只显示了 3 个项目。向下滚动时,第一个将消失,而第四个将出现。第四个项目已经在 Android 系统中,但第五个是从第一个项目创建的,使用相同的视图。就我而言,我有一个带有默认背景的 TextView。但是,如果满足某些条件,我会更改背景。

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    ...
    if (someCondition) {
        holder.textView.setBackground(R.drawable.different_background)
    }
    ...
}

问题是如果不满足条件,我不会设置默认背景(因为我认为视图是为每个项目创建的)。当第一项条件为真时出现问题,因此回收视图与 different_background 一起使用,而不是默认背景。为了避免这个问题,我必须始终设置条件的两个分支并设置默认值,如下所示:

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    ...
    if (someCondition) {
        holder.textView.setBackground(R.drawable.different_background)
    } else {
        holder.textView.setBackground(R.drawable.default_background)
    }
    ...
}
于 2020-02-08T16:05:57.823 回答