Android Fragment之间的数据交流

Android Fragment Argument

众所周知,多用Fragment能打造更灵活的程序。
本文通过一个浅显的例子,来阐释fragment之间基于Argument的数据交流。

简单说一下要实现的目标:
本项目包含两个活动和分别依附于这两个活动的两个Fragment。
简单起见,这里分别为他们起名为:FirstActivityFirstFragmentSecondActivitySecondFragment
他们之间的关系是:
两个活动只负责容纳(或者说托管)其对应的两个Fragment。而具体的显示和与用户交互则由Fragment负责。

为了突出重点,这里只实现最简单的功能:

  • FirstFragment中显示一个ListView,这个ListView显示一串编程语言的名称。
  • 当用户点击其中的item时,会跳转到SecondActivity
  • 这时SecondActivityonCreate()方法启动,在其中加载SecondFragment
  • 最后SecondFragmentTextView控件根据传过来的信息显示相应的编程语言的名字。

如图:


就是这个意思

在代码中实现时,FirstActivitySecondActivity甚至都不需要对应的Layout资源文件。因为它们唯一的作用只是为Fragment提供容器,所以这里只需要在java代码中为两个Activity设置contentView即可:

setContentView(R.layout.common_fragment_container);

这个名为common_fragment_container的布局文件提供了一个FrameLayout来作为Fragment的容器:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/fragment_container"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

根据我们的构想,当用户点击FirstFragment中的ListView的item时,应该跳转到SecondActivity
为此,我们在SecondActivity中定义静态方法:


    private final static String 
    EXTRA_LANGUAGE_PICKED = "language_picked"; //键

    
//静态方法,提供从别的活动跳转到SecondActivity
public static Intent newIntent(Context packageContext, String languagePicked) {
        Intent intent = new Intent(packageContext, SecondActivity.class);
        intent.putExtra(EXTRA_LANGUAGE_PICKED, languagePicked);
        return intent;
    }

FirstFragment中ListView item的点击回调:

public class FirstFragment extends Fragment {

    ListView mList;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.fragment_first, container, false);
        mList = v.findViewById(R.id.list);
//点击回调
        mList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
                Resources resources = getResources();
//得到资源文件中定义的字符串数组
                String[] languages = 
resources.getStringArray(R.array.languages);
                String str = languages[position];
                Intent intent = SecondActivity.newIntent(
getActivity(), str);
//启动SecondActivity
                startActivity(intent);
            }
        });
        return v;

    }
}

FirstFragment通过startActivity(intent)启动SecondActivity之后。
SecondActivity并不直接与用户交互。

它要做的是:

  • 将传入的intent中的用户点击的编程语言名称取出来;
  • 然后传给SecondFragment。由SecondFragment将它显示出来。

SecondFragment想从SecondActivity那儿取到数据有两种方式:

第一种比较直接:

SecondFragment简单粗暴地通过getActivity()方法得到托管自己的SecondActivity
然后通过getIntent()方法得到从FirstFragment中传过来的Intent对象;
最后得到其中的extra信息。

这种方式虽然简单,但也有代价。那就是破坏了封装。使得SecondFragment不能被复用。因为此时它还承担了的工作。

第二种方式比较复杂,但也更灵活:附加argument给Fragment:

要附加argument给Fragment,需要调用Fragment.setArguments(Bundle)方法。而且必须是在fragment创建后,添加给Activity之前。
因此,一般的惯用做法是在Fragment类中添加newInstance()静态方法。
通过这个方法完成fragment实例以及Bundle对象的创建,
最后再把argument放入bundle对象中,并附加给fragment:

//SecondFragment
public class SecondFragment extends Fragment {

private static final String 
ARG_LANGUAGE_PICKED = "arg_language_picked";

    TextView mText;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        
        View v = inflater.inflate(R.layout.fragment_second, container, false);
        mText = v.findViewById(R.id.language_picked);
        String languagePicked 
= getArguments().getString(ARG_LANGUAGE_PICKED);
        mText.setText(languagePicked);
        return v;
    }

//newInstance()方法
    public static Fragment newInstance(String languagePicked) {
        Bundle bundle = new Bundle();
        bundle.putSerializable(ARG_LANGUAGE_PICKED, languagePicked);

        Fragment SecondFragmentInstance = new SecondFragment();
        SecondFragmentInstance.setArguments(bundle);
        return SecondFragmentInstance;
    }

}

现在我们有了这个方法,又得到了FirstFragment传入的Intent对象中的extra信息languagePicked
我们只需要在SecondActivityonCreate()方法中,将languagePicked作为参数传入SecondFragment.newInstance()方法;
即可实现,在SecondFragment创建之后,被添加给SecondActivity之前;
SecondFragment装载argument

//SecondActivity
public class SecondActivity extends AppCompatActivity {

private final static String 
EXTRA_LANGUAGE_PICKED = "language_picked";

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
//使用通用的Fragment容器,
setContentView(R.layout.common_fragment_container);
//因为目前两个Activity的布局中
//其实都只需要一个用于容纳Fragment的frameLayout
       
        //要想在Activity中创建Fragment,先要得到FragmentManager
        FragmentManager fragmentManager = getSupportFragmentManager();
        Fragment fragment = fragmentManager.findFragmentById(R.id.fragment_container);

        if (fragment == null) {
            //在firstActivity中通过Intent跳转到secondActivity,
            //SecondActivity创建之后,从传入的Intent中得到extra信息,
            //然后根据这个信息来创建secondFragment实例,
            //得到的信息将用来作为参数,传入secondFragment的newInstance()方法
            String languagePicked = 
getIntent().getStringExtra(EXTRA_LANGUAGE_PICKED);
//SecondFragment.newInstance()方法
            fragment = SecondFragment.newInstance(languagePicked);

            fragmentManager.beginTransaction()
                    .add(R.id.fragment_container, fragment)
                    .commit();
        }


    }

    //静态方法,提供从别的活动跳转到自身的Intent
    public static Intent newIntent(Context packageContext, String languagePicked) {
        Intent intent = new Intent(packageContext, SecondActivity.class);
        intent.putExtra(EXTRA_LANGUAGE_PICKED, languagePicked);
        return intent;
    }
}

这一做法的灵活之处就在于:
SecondFragment虽然需要得到数据,但是它不再亲自去
而是由托管它的Activity(此处是SecondActivity)来负责提供数据。
如此一来,就实现了SecondActivity的复用。
倘若现在有一个ThirdActivity也想要托管SecondFragment,那它只要能提供数据(类似于SecondActivity提供的languagePicked),那它就一样可以其onCreate()方法中作出类似的实现。
-- end --

水平有限,难免纰漏,如有错误,欢迎指正。
诸君共勉:)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容