ViewPager中片段之间的通信

12
我将尝试完成以下任务:http://android-er.blogspot.com/2012/06/communication-between-fragments-in.html,但是我使用了FragmentStatePagerAdapter。
我有一个Activity,其中包含两个片段(FragmentA和FragmentB)。
FragmentA有一个EditText和一个Button,FragmentB有一个TextView。
现在我想要的是,每当我在EditText中输入内容并单击按钮后,在TextView上就会显示出那个内容。
MainActivity代码:
public class MainActivity extends FragmentActivity {


    ViewPager viewPager = null;
    String TabFragmentB;

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

        viewPager = (ViewPager)findViewById(R.id.pager);    
        FragmentManager fragmentManager = getSupportFragmentManager();
        viewPager.setAdapter(new MyAdapter(fragmentManager));

    }

    public class MyAdapter extends FragmentStatePagerAdapter {  

        public MyAdapter (FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int i) {
            Fragment fragment = null;

            if (i == 0)
            {
                fragment = new FragmentA();
            }
            if (i == 1)
            {
                fragment = new FragmentB();
            }
            return fragment;
        }

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

    public void setTabFragmentB(String t) {
        TabFragmentB = t;   
    }

    public String getTabFragmentB() { 
        return TabFragmentB;
    }

}

碎片A:

public class FragmentA extends Fragment {

    EditText et;
    Button bt;

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

        View v = inflater.inflate(R.layout.fraga, container, false);

        et = (EditText)v.findViewById(R.id.edit1);
        bt = (Button)v.findViewById(R.id.button1);
        bt.setOnClickListener(Click);

        return v;
    }

    OnClickListener Click = new OnClickListener(){

        @Override
        public void onClick(View v) {

            String textPassToB = et.getText().toString();

            String TabOfFragmentB = ((MainActivity)getActivity()).getTabFragmentB();

            FragmentB fragmentB = (FragmentB)getActivity()
               .getSupportFragmentManager()
               .findFragmentByTag(TabOfFragmentB);

            fragmentB.updateText(textPassToB);          
        }   
    };

}

FragmentB:

public class FragmentB extends Fragment {

    TextView tv;

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

        View v = inflater.inflate(R.layout.fragb, container, false);    

        tv = (TextView)v.findViewById(R.id.text1);
        String myTag = getTag();

        ((MainActivity)getActivity()).setTabFragmentB(myTag);

        return v;
    }

    public void updateText(String t){
          tv.setText(t);
         }

}

日志记录器:

FATAL EXCEPTION: main
 java.lang.NullPointerException
        at lmf.sample1.FragmentA$1.onClick(FragmentA.java:43)
        at android.view.View.performClick(View.java:4212)
        at android.view.View$PerformClick.run(View.java:17476)
        at android.os.Handler.handleCallback(Handler.java:800)
        at android.os.Handler.dispatchMessage(Handler.java:100)
        at android.os.Looper.loop(Looper.java:194)
        at android.app.ActivityThread.main(ActivityThread.java:5371)
        at java.lang.reflect.Method.invokeNative(Native Method)
        at java.lang.reflect.Method.invoke(Method.java:525)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:833)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
        at dalvik.system.NativeStart.main(Native Method)

每当我在第一个片段上点击按钮时,我的应用程序就会崩溃。到底是什么问题?
4个回答

30
  1. 您可以使用Intent(在片段B中注册广播接收器并从片段A发送广播)。
  2. 使用EventBus: https://github.com/greenrobot/EventBus 。 这是我最喜欢的方法。 非常方便易用,可实现各个组件之间的简单通信(例如Activity和Services)。

操作步骤:

首先,创建一个代表文本更改事件的类:

public class TextChangedEvent {
  public String newText;
  public TextChangedEvent(String newText) {
      this.newText = newText;
  }
}

然后,在片段A中:

//when text changes
EventBus bus = EventBus.getDefault();
bus.post(new TextChangedEvent(newText));

在片段B中:

EventBus bus = EventBus.getDefault();

//Register to EventBus
@Override
public void onCreate(SavedInstanceState savedState) {
 bus.register(this);
}

//catch Event from fragment A
public void onEvent(TextChangedEvent event) {
 yourTextView.setText(event.newText);
}

公共类TextChangedEvent是否在MainActivity内部? - general_bearbrand
不,它是一个独立的类(当然,如果需要,您可以将其声明为活动的内部类(具有公共静态修饰符),并将其称为MainActivity.TextChangedEvent)。 - Rodion Altshuler
当然,只有在片段B已经被创建时,它才会接收事件。因此,您需要在片段的onCreate()中设置初始文本。 - Rodion Altshuler
最佳方法 - blueware
3
请确保在您想要获取信息的方法上方添加"@Subscribe"。否则它会崩溃。 - Jack
如果带有数据的事件比您的片段文本字段初始化更早,会发生什么?它会崩溃吗? - yozhik

7
片段可以访问它们的父Activity。
因此,最简单的方法是在父Activity中注册回调函数。 更新:已将提交缓存添加到MainActivity。
    public class MainActivity extends FragmentActivity {

        private OnButtonClicked mOnButtonClicked;
        private String mSubmitCache;

        public interface OnButtonClicked {
            void submit(final String s);
        }

        public void setOnButtonClicked(final OnButtonClicked c) {
            mOnButtonClicked = c;
            // deliver cached string, if any
            if (TextUtils.isEmpty(mSubmitCache) == false) {
                c.submit(mSubmitCache);
            }
        }

        public void buttonClicked(final String s)  {
            if (mOnButtonClicked == null) {
                // if FragmentB doesn't exist jet, cache value
                mSubmitCache = s;
                return;
            }
            mOnButtonClicked.submit(s);
        }
    }

    public class FragmentA extends Fragment implements OnClickListener {

        private MainActivity mMain;
        private Button mButton;

        @Override public onAttach(Activity a) {
            mMain = (MainActivity) a;
        }

        @Override public void onClick(View v) {
            mMain.buttonClicked("send this to FragmentB.");
        }
    }

    public class FragmentB extends Fragment implements MainActivity.OnButtonClicked {

        private MainActivity mMain;
        private TextView mTextView;

        // Called when the fragment's activity has been created
        // and this fragment's view hierarchy instantiated
        @Override public void onActivityCreated(Bundle savedState) {
            mMain = (MainActivity) getActivity();
            mMain.setOnButtonClicked(this);
        }

        @Override void submit(final String s) {
            mTextView.setText(s);
        }
    }

我认为这比使用广播或任何第三方库更好! - Slick Slime

3
我使用了Rodion上面的解决方案。但是,除此之外,Android Studio还要求我在onEvent方法之前添加@Subscribe注释。
像这样:

@Subscribe
public void onEvent(TextChangedEvent event) {
    textView.setText(event.newText);
}

根据EventBus' API,订阅者实现事件处理方法(也称为“订阅者方法”),当发布事件时将调用这些方法。这些方法使用@Subscribe注解定义。请注意,使用EventBus 3可以自由选择方法名称(不像EventBus 2中的命名约定)。

1

只有当您切换到FragmentB时,它才会被创建,因此fragmentB.updateText(textPassToB);会产生NullPointerException。

您需要在活动中存储来自EditText的文本,并在稍后(如果)创建FragmentB时需要从中读取值。


我已经编辑了答案,并提出了解决建议。 - Ognyan
你的意思是我会把文本从FragmentA传递到Activity,然后当我在FragmentB时,我会从Activity中获取它? - general_bearbrand
没错。只需在您的活动中为实例变量提供getter和setter,然后在您的片段中使用MyActivity act = (MyActivity) getActivity(); act.getMyVariable()。这不是最干净的方法,但对于像您这样简单的情况是可以接受的。 - Ognyan
只需点击答案左侧的复选标记,它就会变成绿色。 - Ognyan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接