Android中給fragment寫(xiě)入?yún)?shù)的輕量開(kāi)發(fā)包FragmentArgs簡(jiǎn)介
Android開(kāi)發(fā)有時(shí)候會(huì)令人頭痛。你不得不為諸如建立fragment這樣簡(jiǎn)單的事情寫(xiě)很多代碼。幸運(yùn)的是java支持一個(gè)強(qiáng)大的工具:注釋處理器(Annotation Processors)。
Fragment的問(wèn)題是你不得不設(shè)置很多參數(shù),從而讓它正常運(yùn)行。很多android開(kāi)發(fā)新手通常這樣寫(xiě):
public class MyFragment extends Fragment
{
private int id;
private String title;
public static MyFragment newInstance(int id, String title)
{
MyFragment f = new MyFragment();
f.id = id;
f.title = title;
return f;
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState)
{
Toast.makeText(getActivity(), "Hello " + title.substring(0, 3),
Toast.LENGTH_SHORT).show();
}
}
這樣做怎么了?我已經(jīng)在自己的設(shè)備上嘗試過(guò)了,它很好用?
它的卻能工作,但是你有沒(méi)有試過(guò)把你的設(shè)備從豎向改為橫向?你的app將會(huì)因?yàn)镹ullPointerException而崩潰,當(dāng)你試圖訪問(wèn)id或title時(shí)。
我的app是正常的,因?yàn)槲野補(bǔ)pp設(shè)置為豎向。所以我從來(lái)沒(méi)遇到過(guò)這個(gè)問(wèn)題。
隨便你!Android是一個(gè)真正的多任務(wù)操作系統(tǒng)。多個(gè)app在同一時(shí)間運(yùn)行,同時(shí)如果需要內(nèi)存android系統(tǒng)將會(huì)銷(xiāo)毀activity(和其中包含的fragment)??赡苣阍谌粘5腶pp開(kāi)發(fā)中不會(huì)注意這些問(wèn)題。然而,當(dāng)你在play store中發(fā)布后,你將會(huì)注意到你的app崩潰了,但你不知道什么原因。使用你app的用戶可能同時(shí)間使用多個(gè)app,很有可能你的app在后臺(tái)被銷(xiāo)毀了。例如:A 用戶打開(kāi)你的app,MyFragment在屏幕上顯示。下一步你的用戶按了home鍵(這是你的app在后臺(tái)運(yùn)行),并且打開(kāi)了其它應(yīng)用。你的app可能會(huì)因?yàn)獒尫艃?nèi)存而被銷(xiāo)毀。之后,用戶返回你的app,例如通過(guò)多任務(wù)按鈕。所以,Android現(xiàn)在會(huì)怎么做?Android會(huì)恢復(fù)之前的app狀態(tài),同時(shí)恢復(fù)MyFragment,這就是問(wèn)題所在。Fragment試圖訪問(wèn)title,但title是null,因?yàn)樗皇潜挥谰帽4娴摹?/p>
我知道了,所以我需要把它們保存在onSaveInstanceState(Bundle)中?
不是。官方的文檔有一些不清楚,但是onSaveInstanceState(Bundle)的使用方法應(yīng)該跟你用Activity.onSaveInstanceState(Bundle)一樣:你使用這個(gè)方法保存實(shí)例的“臨時(shí)”狀態(tài),例如去處理屏幕的方向(從豎向到橫向,反之亦然)。所以說(shuō)當(dāng)app在后臺(tái)被殺掉時(shí)fragment的實(shí)例狀態(tài)并不能被保存成持久數(shù)據(jù),它的作用是再一次返回前臺(tái)時(shí)恢復(fù)數(shù)據(jù)。它的作用跟Activity.onSaveInstanceState(Bundle)在Activity中的作用相同,它們用于“臨時(shí)”保存實(shí)例狀態(tài)。然而,持久的參數(shù)是通過(guò)intent外部數(shù)據(jù)傳輸?shù)摹?/p>
所以我應(yīng)該在Activity中得Intent保存Fragment的參數(shù)?
不需要,F(xiàn)ragment有它自己的機(jī)制。有兩個(gè)方法:Fragment.setArguments(Bundle)和Fragment.getArguments(),你必須通過(guò)這兩個(gè)方法來(lái)確保參數(shù)被持久保存。這就是我上面提到的痛苦之處。需要有大量的代碼這樣寫(xiě)。第一,你要?jiǎng)?chuàng)建一個(gè)Bundle,然后你需要放入鍵值對(duì),最后調(diào)用Fragment.setArguments()。不幸的是,你的工作還沒(méi)有結(jié)束,你必須通過(guò)Fragment.getArguments()來(lái)讀出Bundle。一些這樣的工作:
public class MyFragment extends Fragment
{
private static String KEY_ID = "key.id";
private static String KEY_TITLE = "key.title";
private int id;
private String title;
public static MyFragment newInstance(int id, String title)
{
MyFragment f = new MyFragment();
Bundle b = new Bundle();
b.putInt(KEY_ID, id);
b.putString(KEY_TITLE, title);
f.setArguments(b);
return f;
}
@Override
public void onCreate(Bundle savedInstanceState)
{
// onCreate it's a good point to read the arguments
Bundle b = getArguments();
this.id = b.getInt(KEY_ID);
this.title = b.getString(KEY_TITLE);
}
@Override
public View onCreate(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState)
{
// No NullPointer here, because onCreate() is called before this
Toast.makeText(getActivity(), "Hello " + title.substring(0, 3),
Toast.LENGTH_SHORT).show();
}
}
我希望你現(xiàn)在能明白我所說(shuō)的“痛苦”。在你的應(yīng)用中你將會(huì)為每一個(gè)fragment寫(xiě)很多代碼。如果有人為你寫(xiě)這樣的代碼,這將不讓人滿意。注釋處理允許你在編譯的時(shí)候生成java代碼。注意我們并不是在討論評(píng)價(jià)在運(yùn)行時(shí)間使用反射的注釋。
FragmentArgs是一個(gè)輕量的包,用于為你的fragment生成精確的java代碼。
import com.hannesdorfmann.fragmentargs.FragmentArgs;
import com.hannesdorfmann.fragmentargs.annotation.Arg;
public class MyFragment extends Fragment
{
@Arg
int id;
@Arg
String title;
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
FragmentArgs.inject(this); // read @Arg fields
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState)
{
Toast.makeText(getActivity(), "Hello " + title, Toast.LENGTH_SHORT)
.show();
}
}
只需要在你的Fragment類(lèi)中加入注釋字段,F(xiàn)ragmentArgs就會(huì)生成引用代碼。在你的Activity中你將使用生成的Builder類(lèi)(你的fragment的后綴是”Builder”),而不是使用new MyFragment()或靜態(tài)的MyFragment.newInstance(int id,String title)方法。
例如:
public class MyActivity extends Activity
{
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
int id = 123;
String title = "test"; // Using the generated Builder
Fragment fragment = new MyFragmentBuilder(id, title).build(); // Fragment Transaction
getFragmentManager().beginTransaction().replace(R.id.container,fragment).commit();
}
}
你可能已經(jīng)注意到在Fragment.onCreate(Bundle)中聲明的FragmentArgs.inject(this)。這個(gè)調(diào)用使你的fragment獲得了生成代碼的連接。你可能會(huì)問(wèn)你自己:“我需不需要在每一個(gè)Fragment中的onCreate(Bundle)中加入inject()方法”。答案是no。你只需要在你的fragment基類(lèi)中插入這一句就可以,并且在所有的fragment中繼承它。
public class BaseFragment extends Fragment
{
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
FragmentArgs.inject(this); // read @Arg fields
}
}
public class MyFragment extends BaseFragment
{
@Arg
String title;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState)
{
Toast.makeText(getActivity(), "Hello " + title, Toast.LENGTH_SHORT)
.show();
}
}
信用:一部分的注釋生成代碼是基于Hugo Visser的Bundles項(xiàng)目。
相關(guān)文章
21天學(xué)習(xí)android開(kāi)發(fā)教程之MediaPlayer
21天學(xué)習(xí)android開(kāi)發(fā)教程之MediaPlayer,MediaPlayer可以播放音頻和視頻,操作相對(duì)簡(jiǎn)單,感興趣的小伙伴們可以參考一下2016-02-02Android基礎(chǔ)之隱藏標(biāo)題欄/設(shè)置為全屏/橫豎屏切換
大家好,本篇文章主要講的是Android基礎(chǔ)之隱藏標(biāo)題欄/設(shè)置為全屏/橫豎屏切換,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12Android項(xiàng)目實(shí)戰(zhàn)教程之高仿網(wǎng)易云音樂(lè)啟動(dòng)頁(yè)實(shí)例代碼
這篇文章主要給大家介紹了關(guān)于Android項(xiàng)目實(shí)戰(zhàn)教程之高仿網(wǎng)易云音樂(lè)啟動(dòng)頁(yè)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-09-09Android編程實(shí)現(xiàn)通訊錄中聯(lián)系人的讀取,查詢,添加功能示例
這篇文章主要介紹了Android編程實(shí)現(xiàn)通訊錄中聯(lián)系人的讀取,查詢,添加功能,涉及Android權(quán)限控制及通訊錄相關(guān)操作技巧,需要的朋友可以參考下2017-07-07Android開(kāi)發(fā)實(shí)現(xiàn)帶清空按鈕的EditText示例
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)帶清空按鈕的EditText,結(jié)合具體實(shí)例形式分析了Android實(shí)現(xiàn)EditText清空按鈕功能相關(guān)操作技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-11-11Android中實(shí)現(xiàn)毛玻璃效果的3種方法
這篇文章主要介紹了Android中實(shí)現(xiàn)毛玻璃效果的3種方法,本文講解了使用系統(tǒng)提供的方法、自定義的方法、C語(yǔ)言實(shí)現(xiàn)方法等3種方法,需要的朋友可以參考下2015-04-04Android 軟鍵盤(pán)狀態(tài)并隱藏輸入法的實(shí)例
這篇文章主要介紹了Android 軟鍵盤(pán)狀態(tài)并隱藏輸入法的實(shí)例的相關(guān)資料,這里提供實(shí)例實(shí)現(xiàn)軟鍵盤(pán)切換并隱藏輸入法的鍵盤(pán),需要的朋友可以參考下2017-09-09Android開(kāi)發(fā)之a(chǎn)ctivity的生命周期詳解
這篇文章主要介紹了Android開(kāi)發(fā)之a(chǎn)ctivity的生命周期,詳細(xì)分析了activity的運(yùn)行原理與生命周期,需要的朋友可以參考下2016-06-06