繁体   English   中英

WorkManager - 当我们同时使用默认初始化和自定义初始化时,我们是否应该删除默认初始化程序?

[英]WorkManager - Should we remove the default initializer, when we use both Default initialization and Custom initialization?

当我将 WorkManager 从“2.2.0”升级到“2.3.0-rc01”时,出现以下新错误

当我导出 APK 时发生错误。

C:\app: Error: Remove androidx.work.impl.WorkManagerInitializer from your AndroidManifest.xml when using on-demand initialization. [RemoveWorkManagerInitializer]

   Explanation for issues of type "RemoveWorkManagerInitializer":
   If an android.app.Application implements
   androidx.work.Configuration.Provider,
   the default androidx.work.impl.WorkManagerInitializer needs to be removed
   from the
   AndroidManifest.xml file.

我不知道为什么我在 2.2.0 中没有收到这样的错误,因为从 2.1.0 开始引入了“按需初始化”。

根据https://developer.android.com/topic/libraries/architecture/workmanager/advanced/custom-configuration#remove-default

我不确定在我的AndroidManifest.xml包含以下内容是否正确。

<provider
    android:name="androidx.work.impl.WorkManagerInitializer"
    android:authorities="${applicationId}.workmanager-init"
    tools:node="remove" />

目前,以下是我的Application类。

MyApplication 类

public class MyApplication extends MultiDexApplication implements Configuration.Provider {
    private static MyApplication me;

    @Override
    public void onCreate() {
        super.onCreate();

        me = this;
    }

    public static MyApplication instance() {
        return me;
    }

    @NonNull
    @Override
    public Configuration getWorkManagerConfiguration() {
        return new Configuration.Builder()
                .build();
    }
}

我如何构建 WorkManager

public static WorkManager getWorkManager() {
    MyApplication myApplication = MyApplication.instance();
    if (myApplication == null) {
        // Very rare edge case. Not sure how it happens. But, it happens :)
        return WorkManager.getInstance();
    } else {
        return WorkManager.getInstance(myApplication);
    }
}

Application类为空时,似乎也很少有机会执行“默认初始化”( WorkManager.getInstance() )。

通过包含以下provider ,我可以轻松消除 APK 导出过程中的错误。 但是,这样做是对的吗?

<provider
    android:name="androidx.work.impl.WorkManagerInitializer"
    android:authorities="${applicationId}.workmanager-init"
    tools:node="remove" />

我们在 WorkManager 2.3.0-*引入了这个 lint 规则。 我们试图通过此 Lint 规则解决的问题是,如果您同时拥有WorkManagerInitializer ContentProvider并且您的Application子类型实现了Configuration.Provider (用于按需初始化) - ContentProvider始终优先。

这可能出乎意料,尤其是当您有额外的Configuration不会生效时,因为ContentProvider始终使用默认配置。

您真正需要做的就是删除默认提供程序。 这样初始化将不再是急切的,而是按需进行的。

是的,如果您想使用按需初始化,您需要删除默认的工作管理器初始化程序,因此请在清单中保留以下代码:

<provider
   android:name="androidx.work.impl.WorkManagerInitializer"
   android:authorities="${applicationId}.workmanager-init"
   tools:node="remove" />

此外,上述文档明确指出您不应调用WorkManager.getInstance() (没有Context参数):

注意:如果在 WorkManager 初始化之前调用已弃用的无参数 WorkManager.getInstance() 方法,该方法将引发异常。 您应该始终使用 WorkManager.getInstance(Context) 方法,即使您没有自定义 WorkManager。

查看 androix/work 更新日志后,您会看到在2.3.0-beta02版本中添加了一个新功能:

  • 添加了 lint 规则,以确保在使用按需初始化时从 AndroidManifest.xml 中删除内容提供程序 androidx.work.impl.WorkManagerInitializer。 ( aosp/1167007 )

从版本 2.2.0 升级到 2.3.0.rc1 后出现此错误的原因是,android 团队添加了一个RemoveWorkManagerInitializerDetector ,它会在构建时抛出您在以下拉取请求中遇到的异常。

现在关于源代码,我建议您将getWorkManager方法直接与应用程序紧密getWorkManager ,如下所示:

import androidx.annotation.NonNull;
import androidx.work.Configuration;
import androidx.work.WorkManager;

public class App extends MultiDexApplication implements Configuration.Provider {
    private static App APP_INSTANCE;

    @Override
    public void onCreate() {
        super.onCreate();
        APP_INSTANCE = this;
    }

    public static App getInstance() {
        return APP_INSTANCE;
    }

    @NonNull
    @Override
    public Configuration getWorkManagerConfiguration() {
        return new Configuration.Builder()
                .build();
    }

    public static WorkManager getWorkManager() {
        return WorkManager.getInstance(APP_INSTANCE);
    }
}

只要您需要在应用程序源代码中调用App.getWorkManager()

如果有的话,您可以为您的 ContentProvider 做一些等效的事情。

PS:有趣的 codelabs 教程退出javakotlin

WorkManagerInitializer用于提供一个上下文WorkManager 发生这种情况是因为内容提供程序在Application之前初始化(请参阅此问题)。 如果您进行自定义初始化,这就是您需要自己提供上下文的原因。 因此,如果您使用自定义初始化,则不需要它。

如果您从内容提供者调用getWorkManager方法,那么您的应用程序实例将为null 要解决此问题,只需通过在内容提供程序中调用getContext将上下文作为参数传递给方法:

public static WorkManager getWorkManager(Context context) {
    return WorkManager.getInstance(context);
}
public class MyContentProvider extends ContentProvider {
    @Override
    public boolean onCreate() {
        WorkManager workManager = getWorkManager(getContext());
        ...
    }
    ...
}

如果您在更新到WorkManager 2.6.0 或更高版本后收到此错误,则必须在清单中使用此代码段:

<provider
    android:name="androidx.startup.InitializationProvider"
    android:authorities="${applicationId}.androidx-startup"
    android:exported="false"
    tools:node="merge">

    <meta-data
        android:name="androidx.work.WorkManagerInitializer"
        android:value="androidx.startup"
        tools:node="remove" />

</provider>

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM