一日一钱,千日一千。绳锯木断,水滴石穿。

PicassoProvider初始化时机

Posted on By TinyVampirePudge

PicassoProvider初始化时机

Picasso版本:

    // picasso
    implementation 'com.squareup.picasso:picasso:2.71828'

在学习Picasso源码的过程中,发现了Picasso对象的初始化不需要传入上下文对象了,示例代码如下:

        // 加载网络图片
        Picasso.get()
                .load("xxx")
                .into(iv1);

我们可以通过Picasso.get()方法直接获取Picasso的对象。

Picasso.get()源码

那么它到底需不需要context对象呢?我们点进去看一下。

  @SuppressLint("StaticFieldLeak") 
  static volatile Picasso singleton = null;
  
  ...
  
  public static Picasso get() {
    if (singleton == null) {
      synchronized (Picasso.class) {
        if (singleton == null) {
          if (PicassoProvider.context == null) {
            throw new IllegalStateException("context == null");
          }
          singleton = new Builder(PicassoProvider.context).build();
        }
      }
    }
    return singleton;
  }

我们可以看到,Picasso使用的是单例模式,是典型的DCL(Double-checked-locking)——双重检查锁模式

Picasso对象singleton的最终初始化时通过给Picasso.Builder构造方法传入一个PicassoProvider.context参数,然后调用Picasso.Builder#build()方法完成Picasso对象的构建。

这说明Picasso还是需要Context对象的,只不过不需要用户传入。

Picasso.Builder

我们看下Picasso.Builder的实现:

    /** Start building a new {@link Picasso} instance. */
    public Builder(@NonNull Context context) {
      if (context == null) {
        throw new IllegalArgumentException("Context must not be null.");
      }
      this.context = context.getApplicationContext();
    }

这里可以明显的看出,我们传入的PicassoProvider.context参数就是Context类型,那么它是何时初始化的呢?

PicassoProvider

我们看下PicassoProvider的实现:

@RestrictTo(LIBRARY)
public final class PicassoProvider extends ContentProvider {

  @SuppressLint("StaticFieldLeak") static Context context;

  @Override public boolean onCreate() {
    context = getContext();
    return true;
  }

  @Nullable @Override
  public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection,
      @Nullable String[] selectionArgs, @Nullable String sortOrder) {
    return null;
  }

  @Nullable @Override public String getType(@NonNull Uri uri) {
    return null;
  }

  @Nullable @Override public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {
    return null;
  }

  @Override public int delete(@NonNull Uri uri, @Nullable String selection,
      @Nullable String[] selectionArgs) {
    return 0;
  }

  @Override
  public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection,
      @Nullable String[] selectionArgs) {
    return 0;
  }
}

PicassoProvider继承自ContentProvider,PicassoProvider唯一的作用是在onCreate方法内部初始化了PicassoProvider.context成员变量,PicassoProvider中的其他方法都是默认实现。

我们知道ContentProvider是在Activity之前启动的,具体调用顺序为Application#attachBaseContext --> ContentProvider#onCreate --> Application#onCreate --> Activity#onCreate,所以我们在Activity中调用Picasso.get时,PicassoProvider.context早已初始化好了,我们就不必担心PicassoProvider.context赋值的问题。

我们知道,在正常使用ContentProvider时,还需要在AndroidManifest.xml中注册。我们找下注册的位置。

很明显,在使用Picasso时我们是不需要在AndroidManifest.xml中注册的,那么PicassoProvider就肯定是注册在picasso的aar文件中的AndroidManifest.xml中了。

这里我们在Project视图下,打开External Libraries目录,找到picasso依赖文件,如下所示:

比较疑惑的是,这里的Gradle依赖的是aar文件,我们应该是可以看到aar文件中的非java文件的,不过这里只显示了classes.jar包。

这样就只能找到我们的aar源文件查看了。右键我们Picasso的Gradle依赖,选择Library Properties...选项,在弹出的对话框中就可以看到aar的源文件了。

我们按图索骥,先找到这个jar文件的目录,目录如下:

虽然找到这个了,不过依然没什么用,我们想要的是aar文件。我们看下上面的文件夹:

这个就是我们要找的aar文件。

查看aar文件的内容

我们先复制一份aar文件出来,然后将后缀改为.zip,接着解压该文件,解压后如下所示:

可以看到我们想找的AndroidManifest.xml,看下内容:

可以看到,PicassoProvider也注册到AndroidManifest.xml中了,至此Picasso中的context的获取原理搞清楚了。

参考

Application, Activity, ContentProvider启动顺序