為什么旋轉(zhuǎn)屏幕后數(shù)據(jù)還在?揭秘ViewModel的生存法則
ViewModel的「退休儀式」
永久退休場(chǎng)景
? 用戶主動(dòng)告別:當(dāng)點(diǎn)擊返回鍵就像和朋友說(shuō)"再見",ViewModel就會(huì)收拾行李離開
? 代碼強(qiáng)制下班:調(diào)用finish()
就像老板說(shuō)"今天不用來(lái)了",ViewModel立刻卷鋪蓋走人
? 系統(tǒng)資源回收:就像突然停電,整個(gè)辦公室(Activity)都被清空
臨時(shí)休假場(chǎng)景
? 屏幕旋轉(zhuǎn):像把手機(jī)轉(zhuǎn)個(gè)方向繼續(xù)用,ViewModel在后臺(tái)淡定喝茶
? 跳轉(zhuǎn)其他頁(yè)面:就像離開客廳去臥室,ViewModel還在客廳等著你回來(lái)
銷毀前的「臨終關(guān)懷」
臨終清理三步走
class GameViewModel : ViewModel() {
// 舉個(gè)栗子:正在進(jìn)行的下載任務(wù)
private var downloadJob: Job? = null
fun startDownload() {
downloadJob = viewModelScope.launch {
// 模擬下載操作
repeat(100) { progress ->
delay(1000)
Log.d("Download", "進(jìn)度:$progress%")
}
}
}
override fun onCleared() {
// 臨終關(guān)懷第一步:取消下載任務(wù)
downloadJob?.cancel()
// 第二步:關(guān)閉數(shù)據(jù)庫(kù)連接
database?.close()
// 第三步:和基類說(shuō)拜拜
super.onCleared()
}
}
代碼小劇場(chǎng)
當(dāng)ViewModel
要退休時(shí),會(huì)自動(dòng)觸發(fā)「臨終關(guān)懷」三部曲:
? 先取消所有正在進(jìn)行的任務(wù)(像收拾辦公桌)
? 關(guān)閉各種連接(像關(guān)電腦拔電源)
? 最后不忘跟上級(jí)(父類)打招呼
實(shí)戰(zhàn)避坑指南
場(chǎng)景 | 數(shù)據(jù)存活? |
屏幕旋轉(zhuǎn) | ? |
返回桌面 | ? |
徹底關(guān)閉頁(yè)面 | ? |
Fragment被移除 | ? |
源碼探秘之旅
// 相當(dāng)于公司HR的離職處理流程
public class ComponentActivity {
protected void onDestroy() {
// 如果不是因?yàn)檠b修(配置變更)要拆除辦公室
if (!isChangingConfigurations()) {
// 開始大掃除
getViewModelStore().clear();
}
}
}
// 清理工具人
public class ViewModelStore {
public final void clear() {
// 給每個(gè)ViewModel發(fā)離職通知
for (ViewModel vm : mMap.values()) {
vm.onCleared(); // 觸發(fā)清理操作
}
// 清空員工名單
mMap.clear();
}
}
解讀:Android
系統(tǒng)就像個(gè)嚴(yán)格的辦公室管理員,在真正要拆除辦公室(Activity銷毀)時(shí),才會(huì)通知所有ViewModel
員工離職,并監(jiān)督他們做好工作交接。
開發(fā)者的生存法則
1. 內(nèi)存泄漏預(yù)防針:在onCleared()
里記得取消網(wǎng)絡(luò)請(qǐng)求和協(xié)程任務(wù),就像出門前要關(guān)煤氣
2. 數(shù)據(jù)保鮮技巧:需要長(zhǎng)期保存的數(shù)據(jù)應(yīng)該存在Repository
層,就像把重要文件存在保險(xiǎn)箱
3. 生命周期觀察員:使用LiveData
自動(dòng)感知生命周期變化,像有個(gè)智能管家?guī)湍闾幚硎聞?wù)
下次當(dāng)你在旋轉(zhuǎn)屏幕時(shí)看到數(shù)據(jù)完好無(wú)損,就知道是ViewModel
在默默守護(hù)啦!