Jetpack系列,字节跳动厂内部超高质量Flutter+Kotlin笔记


class MyViewModel : ViewModel() {



    var handler: Handler = object : Handler() {

        var i = 0

        override fun handleMessage(msg: Message) {

            super.handleMessage(msg)

            print(i++)

            sendEmptyMessageDelayed(0, 500)

        }

    }



    public fun startHandler() {

        handler.sendEmptyMessageDelayed(0, 500)

    }



    override fun onCleared() {

        super.onCleared()

        print("onCleared")

        handler.removeMessages(0)

    }

} 

在MyViewModel中使用Handler创建了一个定时轮训任务,每500毫秒打印一遍日志。接下来,在Activity中创建出ViewModel实例对象,调用其startHandler()方法开始执行:


class ViewModelActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {

        super.onCreate(savedInstanceState)

        setContentView(R.layout.activity_view_model)



        val viewModel = ViewModelProvider(this).get(MyViewModel::class.java)

        viewModel.startHandler()

    }

} 

运行代码会发现,当Activity发生屏幕旋转时,日志并没有中断,且打印出的数字连续,所以证明ViewModel不受Activity的生命周期影响。

上述代码中也介绍了最重要的一个点,那就是ViewModel实例的创建,首先需要创建一个ViewModelProvider类对象,其构造方法中需要传入Activity实例,在androidx中,FragmentActivity默认实现了ViewModelStoreOwner接口:


public class ComponentActivity extends androidx.core.app.ComponentActivity implements

        LifecycleOwner,

        ViewModelStoreOwner,

        HasDefaultViewModelProviderFactory,

        SavedStateRegistryOwner,

        OnBackPressedDispatcherOwner {



} 

创建ViewModelProvider实例后,调用其get()方法就可以:


public class ViewModelStore {



    private final HashMap<String, ViewModel> mMap = new HashMap<>();



    final void put(String key, ViewModel viewModel) {

        ViewModel oldViewModel = mMap.put(key, viewModel);

        if (oldViewModel != null) {

            oldViewModel.onCleared();

        }

    }



    final ViewModel get(String key) {

        return mMap.get(key);

    }



    Set<String> keys() {

        return new HashSet<>(mMap.keySet());

    }



    /**

     *  Clears internal storage and notifies ViewModels that they are no longer used.

     */

    public final void clear() {

        for (ViewModel vm : mMap.values()) {

            vm.clear();

        }

        mMap.clear();

    }

} 

get()方法其核心是在调用ViewModelStore的get()方法,可以看出ViewModelStore类就是使用HashMap将ViewModel与其key(class名称)保存了起来。

其他

需要注意的是,正是由于ViewModel脱离于Activity的生命周期,不建议向ViewModel中传入Context引用,以避免发生内存泄漏。但如果ViewModel中必须要用到Context怎么办呢,可以将ViewModel类继承自AndroidViewModel,AndroidViewModel继承自ViewModel,并且接收Applcation的Context。

ViewModel不会受Actvity生命周期的影响,那是不是可以使用ViewModel替代onSaveInstanceState()方法呢?ViewModel和onSaveInstanceState()还是有不同点的,onSaveInstanceState()方法一般用来存放少量的一些状态数据,并且可以持久化,但ViewModel理论上对数据没有大小限制,但当页面被彻底销毁时,ViewModel中的数据也就不复存在了。

喜欢本文的话,不妨顺手给我点个小赞、评论区留言或者转发支持一下呗😜😜😜~

点击【GitHub】还有彩蛋哦!!!

最后

一线互联网Android面试题含详解(初级到高级专题)

这些题目是今年群友去腾讯、百度、小米、乐视、美团、58、猎豹、360、新浪、搜狐等一线互联网公司面试被问到的题目。并且大多数都整理了答案,熟悉这些知识点会大大增加通过前两轮技术面试的几率

CodeChina开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》

为什么分享?

如果设置门槛,很多开发者朋友会因此错过这套高级架构资料,错过提升成为架构师的可能。这就失去了我们的初衷;让更多人都能通过高效高质量的学习,提升自己的技术和格局,升职加薪。

CodeChina开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》](https://codechina.csdn.net/m0_60958482/android_p7)**

为什么分享?

如果设置门槛,很多开发者朋友会因此错过这套高级架构资料,错过提升成为架构师的可能。这就失去了我们的初衷;让更多人都能通过高效高质量的学习,提升自己的技术和格局,升职加薪。

最后送给大家一句话,望共勉,永远不要放弃自己的梦想和追求;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值