下面是我的代码,其中有3个片段类每个嵌入在ViewPager上的3个选项卡。我有菜单选项。如onOptionsItemSelected()所示,通过选择一个选项,我需要更新当前可见的片段。要更新,我必须调用一个方法,这是在片段类。有人能建议一下如何调用这个方法吗?

public class MainActivity  extends ActionBarActivity {

     ViewPager ViewPager;
     TabsAdapter TabsAdapter;

     @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);

            ViewPager = new ViewPager(this);
            ViewPager.setId(R.id.pager);
            setContentView(ViewPager);

            final ActionBar bar = getSupportActionBar();

            bar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);

            //Attaching the Tabs to the fragment classes and setting the tab title.
            TabsAdapter = new TabsAdapter(this, ViewPager);
            TabsAdapter.addTab(bar.newTab().setText("FragmentClass1"),
                    FragmentClass1.class, null);
            TabsAdapter.addTab(bar.newTab().setText("FragmentClass2"),
              FragmentClass2.class, null);
            TabsAdapter.addTab(bar.newTab().setText("FragmentClass3"),
              FragmentClass3.class, null);


            if (savedInstanceState != null) {
                bar.setSelectedNavigationItem(savedInstanceState.getInt("tab", 0));
            }

        }

        @Override
        public boolean onOptionsItemSelected(MenuItem item) {

            switch (item.getItemId()) {

            case R.id.addText:

           **// Here I need to call the method which exists in the currently visible Fragment class**

                    return true;

            }

            return super.onOptionsItemSelected(item);
        }


     @Override
     protected void onSaveInstanceState(Bundle outState) {
      super.onSaveInstanceState(outState);
            outState.putInt("tab", getSupportActionBar().getSelectedNavigationIndex());

     }

     public static class TabsAdapter extends FragmentPagerAdapter
      implements ActionBar.TabListener, ViewPager.OnPageChangeListener {

      private final Context mContext;
            private final ActionBar mActionBar;
            private final ViewPager mViewPager;
            private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();

            static final class TabInfo {
                private final Class<?> clss;
                private final Bundle args;

                TabInfo(Class<?> _class, Bundle _args) {
                    clss = _class;
                    args = _args;
                }
            }

      public TabsAdapter(ActionBarActivity activity, ViewPager pager) {
       super(activity.getSupportFragmentManager());
                mContext = activity;
                mActionBar = activity.getSupportActionBar();
                mViewPager = pager;
                mViewPager.setAdapter(this);
                mViewPager.setOnPageChangeListener(this);
            }

      public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
                TabInfo info = new TabInfo(clss, args);
                tab.setTag(info);
                tab.setTabListener(this);
                mTabs.add(info);
                mActionBar.addTab(tab);
                notifyDataSetChanged();

            }

      @Override
      public void onPageScrollStateChanged(int state) {
       // TODO Auto-generated method stub

      }

      @Override
      public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
       // TODO Auto-generated method stub

      }

      @Override
      public void onPageSelected(int position) {
       // TODO Auto-generated method stub
       mActionBar.setSelectedNavigationItem(position);
      }

      @Override
      public void onTabReselected(Tab tab, FragmentTransaction ft) {
       // TODO Auto-generated method stub

      }

      @Override
      public void onTabSelected(Tab tab, FragmentTransaction ft) {
       Object tag = tab.getTag();
                for (int i=0; i<mTabs.size(); i++) {
                    if (mTabs.get(i) == tag) {
                        mViewPager.setCurrentItem(i);

                    }
                }

                tabPosition = tab.getPosition();
      }

      @Override
      public void onTabUnselected(Tab tab, FragmentTransaction ft) {
       // TODO Auto-generated method stub

      }

      @Override
      public Fragment getItem(int position) {
       TabInfo info = mTabs.get(position);
                return Fragment.instantiate(mContext, info.clss.getName(), info.args);
      }

      @Override
      public int getCount() {
       return mTabs.size();
      }

     }

    }

假设下面是我想调用的带有updateList()方法的片段类:

 public class FragmentClass1{

    ArrayList<String> originalData;


    @Override
         public View onCreateView(LayoutInflater inflater, ViewGroup container,
           Bundle savedInstanceState) {

          View fragmentView = inflater.inflate(R.layout.frag1, container, false);

          originalData = getOriginalDataFromDB();

          return fragmentView;

         }


    public void updateList(String text)
    {
       originalData.add(text);
       //Here I could do other UI part that need to added
    }
}

当前回答

通过选择一个选项,我需要更新片段 目前可见的。

一个简单的方法是使用一个与FragmentPagerAdapter实现相关的技巧:

case R.id.addText:
     Fragment page = getSupportFragmentManager().findFragmentByTag("android:switcher:" + R.id.pager + ":" + ViewPager.getCurrentItem());
     // based on the current position you can then cast the page to the correct 
     // class and call the method:
     if (ViewPager.getCurrentItem() == 0 && page != null) {
          ((FragmentClass1)page).updateList("new item");     
     } 
return true;

请重新考虑你的变量命名约定,使用类名作为变量名是非常令人困惑的(所以没有ViewPager ViewPager,使用ViewPager mPager为例)。

其他回答

因为FragmentStateAdapter会在FragmentManager中添加新的Fragment,而不是在屏幕旋转时创建新的Fragment,所以你只需要在FragmentManager中获得良好的性能

getActiveFragment() { List<Fragment> fragments = getSupportFragmentManager().getFragments(); for (Fragment Fragment: fragments){ if(MyFragment实例){ 返回(MyFragment); } } 返回null; }

在阅读了所有的评论和答案之后,我将解释这个问题的最佳解决方案。最好的选择是@rik的解决方案,所以我的改进是基于他的。

而不是问每个FragmentClass喜欢

if(FragmentClass1){
   ...
if(FragmentClass2){
   ...
}

创建你自己的接口,让你的子片段实现它,就像这样

public interface MyChildFragment {
    void updateView(int position);
}

然后,您可以启动和更新您的内部片段

Fragment childFragment = (Fragment) mViewPagerDetailsAdapter.instantiateItem(mViewPager,mViewPager.getCurrentItem());

if (childFragment != null) {
   ((MyChildFragment) childFragment).updateView();
}

注:注意你把代码放在哪里,如果你在系统实际创建它之前调用了贪心项目,那么你的子片段的savedInstanceState将为null

public void onCreate(@Nullable Bundle savedInstanceState){
      super(savedInstanceState)
}

会使你的应用程序崩溃。

祝你好运

最好的方法是调用CallingFragmentName fragment = (CallingFragmentName) viewPager .getAdapter() .instantiateItem(viewPager, viewpage . getcurrentitem ());它将重新实例化你所调用的Fragment,这样它就不会抛出空指针异常并调用该Fragment的任何方法。

在我之前的实现中,我存储了一列子片段,以便以后能够访问它们,但这是一个错误的实现,导致了巨大的内存泄漏。

我最终使用实例化项目(…)方法来获得当前的片段:

val currentFragment = adapter?.instantiateItem(viewPager, viewPager.currentItem)

或者获取任何其他碎片位置:

val position = 0
val myFirstFragment: MyFragment? = (adapter?.instantiateItem(viewPager, position) as? MyFragment)

从文档:

为给定位置创建页面。适配器负责 将视图添加到这里给出的容器中,尽管这是必须的 确保在返回时完成 finishUpdate (ViewGroup)。

要获得当前片段-获得ViewPager中的位置在公共void onPageSelected(最终int位置),然后

public PlaceholderFragment getFragmentByPosition(Integer pos){
    for(Fragment f:getChildFragmentManager().getFragments()){
        if(f.getId()==R.viewpager && f.getArguments().getInt("SECTNUM") - 1 == pos) {
            return (PlaceholderFragment) f;
        }
    }
    return null;
}

SECTNUM -在公共静态PlaceholderFragment中分配的位置参数;的片段

getChildFragmentManager()或getFragmentManager() -取决于如何创建SectionsPagerAdapter