同一界面中多个Fragment中的控件ID冲突问题的解决

当一个界面同时含有多个Fragment时,有可能这些Fragment中所对应的布局文件中的控件ID出现相同的情况,这时如果代码处理不当,则会出现某个Fragment中的某操作作用到另外一个Fragment中去了。

例如:某Activity中同时嵌入了两个Fragment,在fragment1.xml和fragment2.xml中都含有id为listView1这个控件

<ListView
    android:id="@+id/listView1"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    />

然后在不同的Fragment所对应的代码中使用如下代码获得相应控件

mListView = (ListView)getActivity().findViewById(R.id.listView1);
这时获得的这个控件对象有可能不是本Fragment中的(如:在fragment1中获得了fragmnet2中的id为R.id.listView1的控件),如此会造成操作错位而形成混乱。究其原因,通过Activity中的findViewById,它会遍历Activity其中嵌入的所有控件,找到第一个符合的即返回,而这个第一个不一定是本Fragmnet中的,这个问题在编译期间是得不到提示的。
为了解决这个问题,或许我们可以修改相应Fragment的布局文件中控件的id标识,使得多个Fragment间的控件id不重复。 但是我们使用Fragment有一个很大的目的是为了代码复用的方便(从一个project转移到另外一个project),而上述的解决方法则显得很不方便。

在Fragment中找到内部的控件

可以在onCreateView中创建Fragment的View时,使用这个View查找控件,代码示例如下:


public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
       View myFragment = inflater.inflate(R.layout.fragment_page2, container, false);
        mListView = (ListView)myFragment.findViewById(R.id.listView1);
        . . .
return myFragment;
}


如果要在Fragment其他函数中要找其中的控件,可以先使用Fragment的getView来先获得View,然后再使用findViewById来找到相应的空间。

在Activity中找到某Fragment中的某控件

从封装的角度来看,一般Activity不要干涉Fragment的内政(呵呵,用上外交语言了),但如果确实需要的话,可以先获得FragmentManager对象(通过getFragmentManager),然后通过在通过FragmentManager中的findFragmentById或者findFragmentByTag来找到相应的Fragment,在使用Fragment的getView,再使用findViewById即可找到相应的控件。


总之,一个Activity中含有Fragment时,一定要注意控件Id的冲突问题,正确处理


这个问题可能是由于使用了`ViewPager`的缓存机制,导致`Fragment`的`View`被销毁后无法重新找到控件解决方案可以尝试以下两种方法: 1. 不使用`ViewPager`的缓存机制 可以通过重写`ViewPager`的`setUserInputEnabled()`方法关闭缓存机制,代码如下: ```java public class NoCacheViewPager extends ViewPager { public NoCacheViewPager(Context context) { super(context); } public NoCacheViewPager(Context context, AttributeSet attrs) { super(context, attrs); } @Override public void setUserInputEnabled(boolean enabled) { super.setUserInputEnabled(enabled); setOffscreenPageLimit(enabled ? 1 : 0); } } ``` 然后在布局文件使用`NoCacheViewPager`替换`ViewPager`即可。 2. 使用`FragmentStatePagerAdapter` `FragmentStatePagerAdapter`是`ViewPager`的一种适配器,它在滑动过程会销毁不需要的`Fragment`,从而避免了`ViewPager`缓存机制带来的问题。代码如下: ```java public class MyPagerAdapter extends FragmentStatePagerAdapter { private List<Fragment> fragments; public MyPagerAdapter(FragmentManager fm, List<Fragment> fragments) { super(fm, FragmentStatePagerAdapter.BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT); this.fragments = fragments; } @Override public Fragment getItem(int position) { return fragments.get(position); } @Override public int getCount() { return fragments.size(); } } ``` 然后在`ViewPager`设置适配器时,使用`MyPagerAdapter`替换原来的适配器即可。 ```java MyPagerAdapter adapter = new MyPagerAdapter(getSupportFragmentManager(), fragments); viewPager.setAdapter(adapter); ``` 希望这些方案对你有所帮助。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值