赞
踩
首先,附上Demo链接: https://github.com/EshelGuo/AppMultipleExtends
众所周知,Java里的继承都是单继承,而我们今天的话题是通过反射实现Application多继承。那么为什么非要Application多继承呢?想必大家在做项目的时候肯定集成很多SDK吧,也应该遇见过一部分SDK要求继承他们提供的Application吧,那假如我集成了两个SDK并且都要求集成他们提供的Application呢?
我首先想到了Application串联继承,自定义Application继承SDK1的Application,然后让SDK1的Application继承SDK2的Application,但是一般三方SDK代码都是封装在jar包里的,想要修改SDK中Application的继承关系不太可能,该方法不可行。
其实最简单的办法就是定义Application,查看SDK1 和SDK2中Application的代码,一般都是在onCreate中有一句或几句初始化SDK的代码,copy粘贴到自己的Application中就好了,但是该方法很容易将SDK初始化逻辑和自己APP初始化逻辑混在一起,加入项目比较大,Application代码比较多,维护起来不方便,并且假如SDK升级修改了初始化接口还要修改Application的代码。当然最重要的是这种方法复制粘贴太 low。哈哈哈~~~
接下来进入正题,由我来介绍一种比较高大上的方法来避免这种尴尬局面。首先看一下Application中的所有用得到的方法:
一般我们用得到的只有 从onCreate()
到 onTrimMemory()
这些生命周期方法和 registerActivityLifecycleCallbacks()
Activity生命周期的注册和取消(Api14 添加的方法)。也就是说,假如我们想要不继承 SDK1Application 就能让这个Application正常使用,首先需要让它的生命周期方法正常运行,其次SDK1Application中的 registerActivityLifecycleCallbacks 能正常使用,还有就是在SDK1Application中使用 this 或者 getApplicationContext() 去开启Activity Toast等必须有效。
要实现最后一点,还需要再看几张图:
首先Application是继承ContextWrapper的,而ContextWrapper继承Context。如下图,ContextWrapper其实就是一个包装类,他的所有实现都在 成员变量 mBase 中,mBase是一个ContextIml类。而今天实现多继承的突破口就在这个mBase上。
假如现在我们不继承SDK的Application,仅仅普通的把他们 new 出来,然后通过反射将自己Application 中的mBase 赋值给 SDK Application中的mBase,并且在自己的Application生命周期中调用SDKApplication的生命周期方法,那么SDK 不就相当于间接继承了我们的Application了吗?
下面只说思路,想要看代码的话请到最上边github链接下载Demo。
attachBaseContext ()
中创建 SDKApplication 并通过反射调用 它的 attachBaseContext()
方法。Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。