當我們在使用第三方庫耙考,或者自己封裝庫,如果需要需要用到Context時错览,一般做法就是將初始化方法暴露給調(diào)用方,讓調(diào)用方在初始化類庫時煌往,傳入Context倾哺。
publi class App extends Application {
/**
* 是否是Debug環(huán)境
*/
public static final boolean IS_DEBUG = true;
private static App mContext;
@Override
public void onCreate() {
super.onCreate();
//ARouter初始化
if (IS_DEBUG) {
ARouter.openLog();
ARouter.openDebug();
}
ARouter.init(this);
}
public static Context getInstance() {
return mContext;
}
}
解決方案
其實無侵入式獲取Context的實現(xiàn)很簡單,就是使用一個ContentProvider刽脖,ContentProvider的onCreate()方法調(diào)用時羞海,調(diào)用getContext()即可獲取到Context,再靜態(tài)變量保存曲管,后續(xù)直接獲取即可却邓。
Picasso的無侵入式獲取Context
上述的原理,其實是從Picasso中借鑒的院水,一起來看一下吧腊徙。
- Picasso實例獲取。重點在Double-Check單例中的PicassoProvider.context檬某,調(diào)用PicassoProvider的context屬性獲取Context撬腾。
static volatile Picasso singleton = null;
public static Picasso get() {
if (singleton == null) {
synchronized (Picasso.class) {
if (singleton == null) {
//重點在這里,PicassoProvider.context獲取Context
if (PicassoProvider.context == null) {
throw new IllegalStateException("context == null");
}
singleton = new Builder(PicassoProvider.context).build();
}
}
}
return singleton;
}
- PicassoProvider類
PicassoProvider其實是一個ContentProvider的子類恢恼,只要將PicassoProvider在AndroidManifest清單文件中注冊民傻,啟動App時,Android框架會初始化這個PicassoProvider场斑,onCreate()方法被調(diào)用時漓踢,getContext()保存Context即可。
public final class PicassoProvider extends ContentProvider {
@SuppressLint("StaticFieldLeak") static Context context;
@Override public boolean onCreate() {
context = getContext();
return true;
}
}
代碼實現(xiàn)
- ApplicationContextProvider漏隐,和Picasso一樣喧半,是ContentProvider的子類。在onCreate()中獲取到Context青责,再保存到ContextProvider實例中挺据。
public class ApplicationContextProvider extends ContentProvider {
@SuppressLint("StaticFieldLeak")
static Context mContext;
@Override
public boolean onCreate() {
mContext = getContext();
return false;
}
//...省略其他必須復(fù)寫的方法(空實現(xiàn)即可)
}
- 在AndroidManifest.xml中注冊ApplicationContextProvider
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.app.contextprovider">
<application>
<!-- 全局Context提供者 -->
<provider
android:name=".ApplicationContextProvider"
android:authorities="${applicationId}.contextprovider"
android:exported="false" />
</application>
</manifest>
- ContextProvider取具,提供全局Context的單例類,提供get()方法獲取單例實例吴菠,第一次構(gòu)造時才從ApplicationContextProvider中獲取Context來初始化自身者填。
public class ContextProvider {
@SuppressLint("StaticFieldLeak")
private static volatile ContextProvider instance;
private Context mContext;
private ContextProvider(Context context) {
mContext = context;
}
/**
* 獲取實例
*/
public static ContextProvider get() {
if (instance == null) {
synchronized (ContextProvider.class) {
if (instance == null) {
Context context = ApplicationContextProvider.mContext;
if (context == null) {
throw new IllegalStateException("context == null");
}
instance = new ContextProvider(context);
}
}
}
return instance;
}
/**
* 獲取上下文
*/
public Context getContext() {
return mContext;
}
public Application getApplication() {
return (Application) mContext.getApplicationContext();
}
}
- 使用,例如使用全局Context發(fā)送廣播
Intent intent = new Intent(action_update_user_info)
ContextProvider.get().getContext().sendBroadcast(intent);
leakcanary中的無侵入式初始化
leakcanary是Android非常出名的內(nèi)存泄露檢測庫做葵,由square公司出品占哟。以前版本的需要我們手動調(diào)用install()初始化,最新版已經(jīng)不需要了酿矢,原因就是也采用ContentProvider自動被Android框架調(diào)用的方式榨乎,獲取到Context后進行install()。
AppWatcherInstaller瘫筐,也是ContentProvider的子類蜜暑。在onCreate()時,調(diào)用了InternalAppWatcher的install()方法進行安裝策肝。
//ContentProvider子類
internal class AppWatcherInstaller : ContentProvider() {
override fun onCreate(): Boolean {
SharkLog.logger = DefaultCanaryLog()
val application = context!!.applicationContext as Application
//無侵入式安裝
InternalAppWatcher.install(application)
return true
}
//省略其他代碼...
}
internal object InternalAppWatcher {
//省略其他代碼...
//安裝
fun install(application: Application) {
SharkLog.d { "Installing AppWatcher" }
checkMainThread()
//安裝過就不安裝了肛捍,避免多進程重復(fù)初始化
if (this::application.isInitialized) {
return
}
InternalAppWatcher.application = application
//安裝
val configProvider = { AppWatcher.config }
ActivityDestroyWatcher.install(application, objectWatcher, configProvider)
FragmentDestroyWatcher.install(application, objectWatcher, configProvider)
onAppWatcherInstalled(application)
}
//省略其他代碼...
}
AutoSize屏幕適配框架自動初始化
AutoSize是一個今日頭條屏幕適配方案的第三方庫,內(nèi)部也有一個InitProvider之众,繼承于ContentProvider拙毫,在onCreate()中調(diào)用AutoSizeConfig類的init()方法進行初始化,減少了配置棺禾。
public class InitProvider extends ContentProvider {
@Override
public boolean onCreate() {
AutoSizeConfig.getInstance()
.setLog(true)
.init((Application) getContext().getApplicationContext())
.setUseDeviceSize(false);
return true;
}
//省略其他代碼...
}
總結(jié)
ContentProvider方法無侵入式初始化方案的優(yōu)缺點:
優(yōu)點:對于固定的初始化配置缀蹄,可以使用ContextProvider方案減少調(diào)用方的配置,減少出錯膘婶。
缺點:如果初始化非常耗時缺前,無疑會拖慢App的啟動,如果是耗時初始化悬襟,應(yīng)該提供給調(diào)用方自行決定衅码,例如將初始化推遲到主界面onCreate()時才調(diào)用初始化。