从片段调用活动方法

Posted

技术标签:

【中文标题】从片段调用活动方法【英文标题】:Call an activity method from a fragment 【发布时间】:2012-09-21 12:05:18 【问题描述】:

尝试从片段调用我的活动中的方法。我希望片段提供方法数据并在方法返回时获取数据。我想实现类似于调用静态方法,但不使用静态,因为它会在活动中产生问题。

对 Fragment 不熟悉,所以我需要一个简单的教学解释!

谢谢!

【问题讨论】:

【参考方案1】:

从片段到活动:

((YourActivityClassName)getActivity()).yourPublicMethod();

从活动到片段:

FragmentManager fm = getSupportFragmentManager();

//if you added fragment via layout xml
YourFragmentClass fragment = (YourFragmentClass)fm.findFragmentById(R.id.your_fragment_id);
fragment.yourPublicMethod();

如果您通过代码添加片段并在添加片段时使用tag 字符串,请改用findFragmentByTag

YourFragmentClass fragment = (YourFragmentClass)fm.findFragmentByTag("yourTag");

【讨论】:

要小心,因为如果演员表不起作用会发生意想不到的事情......:S 为了避免演员阵容问题,请使用:Activity act = getActivity(); if (act instanceof YourActivityClassName) ((YourActivityClassName) act).yourPublicMethod(); 投射 Activity 是糟糕的设计和不安全的。 Fragment 不限于特定的 Activity。 Fragments 意味着可以在任何 Activity 中重复使用和设置。如果我有 5 个活动使用相同的片段怎么办? Marco 的答案是正确的,也是片段间和 Activity-Fragment 通信的良好实践。 @Kay 不一定。片段可以用作任何更大活动的“片段”。例如,创建响应式 UI。我很少使用相同的片段并将其附加到不同的活动主机。【参考方案2】:

您可能应该尝试将片段与活动分离,以防您想在其他地方使用它。您可以通过创建您的活动实现的接口来做到这一点。

所以你会定义一个如下所示的接口:

例如,假设您想给活动一个字符串并让它返回一个整数:

public interface MyStringListener
    public Integer computeSomething(String myString);

这可以在片段或单独的文件中定义。

然后您将让您的活动实现该接口。

public class MyActivity extends FragmentActivity implements MyStringListener

  @Override
  public Integer computeSomething(String myString)
   /** Do something with the string and return your Integer instead of 0 **/ 
   return 0;
  


然后在您的片段中,您将有一个 MyStringListener 变量,您将在片段的 onAttach(Activity activity) 方法中设置侦听器。

public class MyFragment 

        private MyStringListener listener;

        @Override
        public void onAttach(Context context) 
            super.onAttach(context);
            try 
                listener = (MyStringListener) context;
             catch (ClassCastException castException) 
                /** The activity does not implement the listener. */
            
        

    

编辑(17.12.2015):onAttach(Activity activity) is deprecated, use onAttach(Context context) instead, it works as intended

第一个答案肯定有效,但它将您当前的片段与主机活动结合起来。如果您想在另一个活动中使用它,将片段与主机活动分离,这是一种很好的做法。

【讨论】:

对于其他人来说,虽然公认的答案显然确实有效,但从设计的角度来看,这是更好、更安全的方法。 这个答案在代码设计方面要好得多。如果 Activity 被错误地投射,它也不会导致崩溃 +1 但我不会在 onAttach 中使用 try-catch。让它失败。如果监听器是可选的(也就是说,失败是不合适的),在片段中添加一个 set/addListener 方法。 对面通讯请见:developer.android.com/training/basics/fragments/…。使用片段的接口(这也是执行片段->活动通信的安全方法,如上所述),如果您想根据片段->活动采取行动,您也可以从您的活动中调用片段中的方法通讯。 我认为 MyFragment 覆盖 onDetach 并将侦听器设置为 null 以避免内存泄漏也很重要。【参考方案3】:

对于 Kotlin 开发人员

(activity as YourActivityClassName).methodName()

对于 Java 开发人员

((YourActivityClassName) getActivity()).methodName();

【讨论】:

如果我们运行这段代码,它会在 kotlin 中出错。还有其他方法吗? 当我运行它时。我得到 ActivityClass 的空值,我认为这不是在 kotlin 中执行此操作的正确方法,即使没有错误。或者可能是一个错误? @JakeGarbo 正确的方式,否则 12 人没有投票给它。第二件事有时 getActivity() 返回 null 检查那些关于 SO 的问题。【参考方案4】:

在我了解更多片段的工作原理后更新。每个片段都属于一个父活动。所以只需使用:

getActivity().whatever

从片段内部。这是一个更好的答案,因为您避免了多余的演员表。如果您无法使用此解决方案避免演员阵容,请使用以下解决方案。

============

你要做的就是投射到外部活动

((MainActivity) getActivity()).Method();

创建一个新实例会混淆 android 框架并且它无法识别它。 另见:

https://***.com/a/12014834/1984636

https://***.com/a/2042829/1984636

【讨论】:

【参考方案5】:

虽然我完全喜欢 Marco 的回答,但我认为公平地指出,您也可以使用基于发布/订阅的框架来实现相同的结果,例如,如果您使用事件总线,您可以执行以下操作

片段:

EventBus.getDefault().post(new DoSomeActionEvent()); 

活动:

 @Subscribe
onSomeActionEventRecieved(DoSomeActionEvent doSomeActionEvent)
//Do something


【讨论】:

【参考方案6】:

在 kotlin 中,您可以从片段中调用活动方法,如下所示:

var mainActivity: MainActivity = activity as MainActivity
        mainActivity.showToast() //Calling show toast method of activity

【讨论】:

【参考方案7】:

感谢@BIJAY_JHA 和@Manaus。我使用 Kotlin 版本来调用我的 signIn() 方法,该方法位于 Activity 中并且我从 Fragment 调用。我在 Android 中使用导航架构,因此侦听器接口模式不在片段中:

 (activity as MainActivity).signIn() 

【讨论】:

【参考方案8】:

我是这样做的:

首先制作界面

interface NavigationInterface 
    fun closeActivity()

接下来确保活动实现接口并覆盖接口方法

class NotesActivity : AppCompatActivity(), NavigationInterface 

    override fun onCreate(savedInstanceState: Bundle?) 
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_notes)
        setSupportActionBar(findViewById(R.id.toolbar))
    

    override fun closeActivity() 
        this.finish()
    

然后确保在片段中创建接口监听器

private lateinit var navigationInterface: NavigationInterface

override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
): View? 
    //establish interface communication
    activity?.let 
        instantiateNavigationInterface(it)
    
    // Inflate the layout for this fragment
    return inflater.inflate(R.layout.fragment_notes_info, container, false)


private fun instantiateNavigationInterface(context: FragmentActivity) 
    navigationInterface = context as NavigationInterface

然后你可以像这样拨打电话:

view.findViewById<Button>(R.id.button_second).setOnClickListener 
    navigationInterface.closeActivity()

【讨论】:

【参考方案9】:

要通过片段访问在 Activity 中声明的函数,请使用接口,如 marco 的回答所示。

如果您没有标签或 id,则可以通过您的 Activity 访问在 Fragment 中声明的函数

private void setupViewPager(ViewPager viewPager) 
    //fragmentOne,fragmentTwo and fragmentThree are all global variables
    fragmentOne= new FragmentOne();
    fragmentTwo= new FragmentTwo();
    fragmentThree = new FragmentThree();

    viewPagerAdapteradapter = new ViewPagerAdapter(getSupportFragmentManager());
    viewPagerAdapteradapter.addFragment(fragmentOne, "Frag1");
    viewPagerAdapteradapter.addFragment(fragmentTwo, "Frag2");
    viewPagerAdapteradapter.addFragment(fragmentThree, "Frag3");

    //viewPager has to be instantiated when you create the activity:
    //ViewPager viewPager = (ViewPager)findViewById(R.id.pager);
    //setupViewPager(viewPager);
    //Where R.id.pager is the id of the viewPager defined in your activity's xml page.

    viewPager.setAdapter(viewPagerAdapteradapter);


    //frag1 and frag2 are also global variables
    frag1 = (FragmentOne)viewPagerAdapteradapter.mFragmentList.get(0);
    frag2 = (FragmentTwo)viewPagerAdapteradapter.mFragmentList.get(1);;


    //You can use the variable fragmentOne or frag1 to access functions declared in FragmentOne



这是 ViewpagerAdapterClass

    class ViewPagerAdapter extends FragmentPagerAdapter 
    public final List<Fragment> mFragmentList = new ArrayList<>();
    private final List<String> mFragmentTitleList = new ArrayList<>();

    public ViewPagerAdapter(FragmentManager manager) 
        super(manager);
    

    @Override
    public Fragment getItem(int position) 
        return mFragmentList.get(position);
    

    @Override
    public int getCount() 
        return mFragmentList.size();
    

    public void addFragment(Fragment fragment, String title) 
        mFragmentList.add(fragment);
        mFragmentTitleList.add(title);
    

    @Override
    public CharSequence getPageTitle(int position) 
        return mFragmentTitleList.get(position);
    

这个答案适合像我这样的菜鸟。祝你有美好的一天。

【讨论】:

【参考方案10】:

这是来自 Fragment 类...

((KidsStoryDashboard)getActivity()).values(title_txt,bannerImgUrl);

来自活动类的代码...

 public void values(String title_txts, String bannerImgUrl) 
    if (!title_txts.isEmpty()) 

//Do something to set text 
    
    imageLoader.displayImage(bannerImgUrl, htab_header_image, doption);

【讨论】:

【参考方案11】:

我一直在寻找最好的方法来做到这一点,因为并非我们要调用的每个方法都位于具有相同 Activity Parent 的 Fragment 中。

在你的片段中

public void methodExemple(View view)

        // your code here

        Toast.makeText(view.getContext(), "Clicked clicked",Toast.LENGTH_LONG).show();
    

在你的活动中

new ExempleFragment().methodExemple(context); 

【讨论】:

【参考方案12】:

我已经尝试了这个线程中显示的所有方法,但没有一个对我有用,试试这个。它对我有用。

((MainActivity) getContext().getApplicationContext()).Method();

【讨论】:

【参考方案13】:
((your_activity) getActivity).method_name()

your_activity 是您的活动名称,method_name() 是您要调用的方法的名称。

【讨论】:

【参考方案14】:

Kotlin 试试

class DataForm : Fragment() 
    override fun onViewCreated(view: View, savedInstanceState: Bundle?) 
        Tasks(this).getData()
    

    fun getResponse(response: String) 
        // code
    


class Tasks(private val context: Any) 
    fun getData() 

        val getContext = (context as DataForm).activity
        val getFragment = (context as DataForm)

        val responseListener = Response.Listener<String>  response ->
            getFragment.getResponse(response)
        

        val errorListener = Response.ErrorListener  error ->
            error.printStackTrace();
        

        val stringRequest = StringRequest(Request.Method.GET, url, responseListener, errorListener)
        Volley.newRequestQueue(getContext).add(stringRequest)
    

【讨论】:

我喜欢这个,因为它消除了孩子对父母的知识水平。片段不应该直接调用父方法。【参考方案15】:

((YourActivityName)getActivity()).functionName();

例如:((SessionActivity)getActivity()).changeFragment();

注意:类名应该是公开的

【讨论】:

【参考方案16】:

从片段到活动:

((YourActivityClassName)requireActivity()).yourPublicMethod();

【讨论】:

【参考方案17】:

从各自的片段中调用活动方法的最佳方式

(activity as YourActivity).activtiyMethod()

从您的活动中使用此行。例如

假设您有 Activity A 和方法 add() 以及您的片段 ABC,并且您想从片段 ABC 调用方法 add,

(activity as A).add()

【讨论】:

以上是关于从片段调用活动方法的主要内容,如果未能解决你的问题,请参考以下文章

如何从主要活动中调用片段方法

从活动中调用片段方法

从活动调用片段方法时出现空指针异常

从片段中调用父母的活动

从意图活动访问片段方法

从活动中调用片段中的非静态方法?