OnClick回调已销毁的活动称为

时间:2017-09-17 17:01:56

标签: android android-recyclerview onclicklistener activity-lifecycle

首先 - 背景:

在我的应用程序中,我有两个非常相似的活动。它们都是通过RecyclerView显示的相同卡片的列表。其中一个是onClick / onLongClick侦听器,缺少上下文操作模式。

我已经通过制作两个子类来实现它们,其中一个父类包含所有公共代码,两个抽象方法用于设置onClick和onLongClick侦听器。

因此,两个活动都使用相同的数据和资源,主要区别在于可能的行动集。

问题:

启动了儿童A.来自其他活动的地方被称为我有可能执行子B.这一切都按预期运作:

A - > X - >乙

当我使用后退按钮从孩子B导航回到孩子A时出现问题:

A< - X < - B

在调试器中我看到了 RecyclerView中的部分卡属于此时已被销毁的活动B.由于两项活动都显示相同的数据 -  用户界面看起来很好,但这个逻辑结果是从活动A调用活动B的onClick回调,这不是我想要的。

代码:

父类构造函数和回调:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
    setSupportActionBar(toolbar);

    dbClass_ = new DBClass(this);

    rv_ = (RecyclerView) findViewById(R.id.entity_list);
    rv_.setLayoutManager(new LinearLayoutManager(this));

    ListedEntityAdapter entityAdapter = new ListedEntityAdapter(this, dbClass_, this, this);
    rv_.setAdapter(entityAdapter);
}

@Override
abstract public void onClick(View view);
@Override
abstract public boolean onLongClick(View view);

xml中的Recyclerview:

<android.support.v7.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    app:layout_behavior="@string/appbar_scrolling_view_behavior"
    android:id="@+id/entity_list"
    android:scrollbars="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:showIn="@layout/activity_main">

A类的OnClick侦听器:

@Override
public void onClick(View view) {
    if ( contextHandler_.isActive() )
    {
        contextHandler_.setSelected( view );
    } else {
        TextView tv = (TextView)view.findViewById( R.id.listed_entity_name );
        ShowDetailsActivity( (Long) tv.getTag( R.string.item_id_tag ), (Integer)tv.getTag( R.string.item_pos_tag ));
    }
}

@Override
public boolean onLongClick(View view) {
    startActionMode(contextHandler_);
    contextHandler_.setSelected( view );
    return true;
}

ShowDetailsActivity可以调用活动B:

@Override
public void onClick(View view) {
    int id  = view.getId();
    switch ( id )
    {
        [...]
        case R.id.item_path:
            Intent itemList = new Intent( this, B.class );
            startActivityForResult( itemList, B.CONTAINER_CHOICE );
            break;
        [...]
    }

最后B类有自己的回调,在这个阶段什么都不做:

@Override
public void onClick(View view) {
    return;
}

@Override
public boolean onLongClick(View view) {
    return true;
}

A和B都没有定义构造函数(它们使用父类之一)。 有人可以解释一下导致这种行为的原因吗?

1 个答案:

答案 0 :(得分:0)

似乎我的问题的答案是在代码中没有在这里发布。 在Web ViewHolder中可用的教程中,经常被声明为静态,我也是这样做的。这是我的错误。

A和B具有相同适配器的不同实例,它们引用来自调用者的onClick回调。由于此引用是静态的 - 回调对于所有实例都是通用的:

class ListedEntityAdapter extends 
RecyclerView.Adapter<ListedEntityAdapter.ViewHolder> {
    private static View.OnLongClickListener onLongCardClick;
    private static View.OnClickListener onCardClick;
    [...]
    public ListedEntityAdapter(Context ctx, DBClass db, View.OnClickListener onClick, View.OnLongClickListener onLongClick ) { 
        onCardClick = onClick;
        onLongCardClick = onLongClick;
        [...]
    }

    public static class ViewHolder extends RecyclerView.ViewHolder {
        [...]
        private ViewHolder(View itemView) {
            super(itemView);
            itemView.setOnLongClickListener( onLongCardClick );
            itemView.setOnClickListener( onCardClick );
            [...]
        }
    }
}

结果发生了什么:

1)创建并启动的适配器,参考A的回调进行初始化。

2)B活动创建并稍后启动,适配器初始化时参考B的回调。

3)我们返回A.由于没有调用onCreate of A,因此没有重新创建适配器,它只有一个注释:静态 onClick回调在B创建后不会更改。这就是为什么不同的适配器实例使用相同的回调。

通过从ViewHolder中移除'static'并在适配器中回调成员来修复相当愚蠢的错误。