問題引入
? ? ? ? 在項(xiàng)目某個(gè)Activity中使用viewPage下嵌套了3個(gè)fragment A析既,B,C供鸠。A頁面保存一些需要聯(lián)網(wǎng)獲取但是在每次使用過程中不會(huì)改變的信息锌钮。?測試軟件時(shí)發(fā)現(xiàn)每次打開c?fragment后再點(diǎn)開a?fragment?a中的view需要重新聯(lián)網(wǎng)獲取數(shù)據(jù)。
問題分析
? ? ? ?查詢資料可知viewpage會(huì)緩存當(dāng)前fragment左邊和右邊總共3個(gè)頁面,當(dāng)我打開c fragment時(shí)候,viepage主動(dòng)殺死?a頁面
解決方案一
? ? ? ? 立刻想到,重寫fragment的onSaveInstanceState() 和onCreateView() ,在fragment被銷毀是將view?的數(shù)據(jù)存起來,在重新打開頁面時(shí)候自動(dòng)填寫.
? ? ? ? 結(jié)果:失敗!
? ? ? ? 給方法打上log,運(yùn)行app
? ? ? ?觀察發(fā)現(xiàn)在打開fragment?c?時(shí)候?并沒有調(diào)用?fragment?a?的onSaveInstanceState()而是直接調(diào)用了?ondestroyview(),
? ? ? ?原因是系統(tǒng)在調(diào)用onPause?和?onStop時(shí)候并不是每次都會(huì)調(diào)用onSaveInstanceState,只有系統(tǒng)自動(dòng)回收資源的時(shí)候覺得該 Activit 在將來某個(gè)時(shí)刻重新回來時(shí)可以恢復(fù)其之前的狀態(tài)扑浸。例如烧给,如果 Activity B 啟用后位于 Activity A 的前端,在某個(gè)時(shí)刻 Activity A 因?yàn)橄到y(tǒng)回收資源的原因要被銷毀首装,Activity A 有機(jī)會(huì)通過 onSaveInstanceState() 來保存其用戶界面狀態(tài)创夜,使得將來用戶返回到 Activity A 的時(shí)候能夠通過onCreate(Bundle) 或者onRestoreInstanceState(Bundle) 來恢復(fù)其界面狀態(tài)。而viewPage直接調(diào)用了fragment?的?ondestroyview仙逻。而并沒有調(diào)用ondeatch所以fragment是還存的,只是里面的view?被摧毀了,所以下次點(diǎn)開fragment a?需要重新獲取數(shù)據(jù).
解決方案二
? ? ? ? 因?yàn)閕nfoactivity?和?fragment?a?其實(shí)是一直存活的沒有被殺死?所以我們不能通過他們的生命周期毀回調(diào)函數(shù)來實(shí)現(xiàn)自動(dòng)填充view這個(gè)功能,難道我們就束手無策了嘛?
? ? ? ? 顯然不是,其實(shí)view是有自己的生命周期的,view?也有自己的onSaveI nstanceState?和?onRestoreInstanceState函數(shù),我們只要在view中重寫這兩個(gè)函數(shù)就可以實(shí)現(xiàn)以上功能
? ? ? ? 而且更加好的消息是基本上驰吓,每個(gè)標(biāo)準(zhǔn)視圖(例如EditText,TextView系奉,Checkbox等)都已經(jīng)在內(nèi)部實(shí)現(xiàn)了這些功能檬贰,你只需要將其設(shè)置android:freezeText為true。
? ? ? ? 但是某些自定義的view未必實(shí)現(xiàn)這個(gè)功能,到時(shí)候就需要我們自己重寫了.
?總結(jié)
? ? ? ? 用onSaveI nstanceState?和?onRestoreInstanceState函數(shù)是非常好的用來保護(hù)和自動(dòng)填充內(nèi)容的方式,但是一定要分清楚avitvity ,fragment ,和view各自的生命周期,如果任何屬性屬于View缺亮,請?jiān)赩iew內(nèi)部進(jìn)行狀態(tài)保存/還原,不要將view?的屬性放在fragment?的生命周期中操作