首先 - 背景:
在我的应用程序中,我有两个非常相似的活动。它们都是通过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都没有定义构造函数(它们使用父类之一)。 有人可以解释一下导致这种行为的原因吗?
答案 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'并在适配器中回调成员来修复相当愚蠢的错误。