多进程Android Pie(9.0)WebView

Lio*_*luz 17 android multiprocessing android-service android-9.0-pie

从Android Pie(API 28)开始,Google不允许在2个不同的进程中使用单个WebView实例。

说明文件: https //developer.android.com/reference/android/webkit/WebView.html#setDataDirectorySuffix(java.lang.String)

根据要求,我打电话给我,WebView.setDataDirectorySuffix("dir_name_no_separator")但不幸的是,我得到了一个例外。我试图在第二个进程Service onCreate()中调用此方法。

java.lang.RuntimeException: Unable to create service com.myapp.service.MyService: java.lang.IllegalStateException: Can't set data directory suffix: WebView already initialized
        at android.app.ActivityThread.handleCreateService(ActivityThread.java:3544)
        at android.app.ActivityThread.access$1300(ActivityThread.java:199)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1666)
        at android.os.Handler.dispatchMessage(Handler.java:106)
        at android.os.Looper.loop(Looper.java:193)
        at android.app.ActivityThread.main(ActivityThread.java:6669)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)
     Caused by: java.lang.IllegalStateException: Can't set data directory suffix: WebView already initialized
        at android.webkit.WebViewFactory.setDataDirectorySuffix(WebViewFactory.java:136)
        at android.webkit.WebView.setDataDirectorySuffix(WebView.java:2165)
        at com.myapp.service.MyService.onCreate(MyService.java:134)
Run Code Online (Sandbox Code Playgroud)

我找不到该异常的任何原因。我没有两次调用此方法,也没有在主过程中调用它。有任何想法吗?

Lio*_*luz 14

解决了。

我的项目托管AdMob广告,我MobileAds.initialize()Application类中调用方法onCreate()。广告初始化程序会加载WebView,现在您将无法在调用该WebView.setDataDirectorySuffix("dir_name_no_separator")方法之前在新流程中执行此操作。

创建第二个进程时,它还会经历相同的应用程序创建流程,这意味着它onCreate()Application类内部调用相同的流程,该类调用 MobileAds.initialize()试图创建新WebView实例的,从而导致崩溃。

IllegalStateException: Can't set data directory suffix: WebView already initialized
Run Code Online (Sandbox Code Playgroud)

我该如何解决?

我使用以下方法获取进程名称,并检查它是否是我的主进程-调用MobileAds.initialize()方法,如果这是我的第二个进程,则调用该 WebView.setDataDirectorySuffix("dir_name_no_separator")方法。

获取进程名称:

public static String getProcessName(Context context) {
    ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
    for (ActivityManager.RunningAppProcessInfo processInfo : manager.getRunningAppProcesses()) {
        if (processInfo.pid == android.os.Process.myPid()) {
            return processInfo.processName;
        }
    }

    return null;
}
Run Code Online (Sandbox Code Playgroud)

应用程序类onCreate():

if (!Utils.getProcessName(this).equals("YOUR_SECOND_PROCESS_NAME")) {
    MobileAds.initialize(this);
} else {
    WebView.setDataDirectorySuffix("dir_name_no_separator")
}
Run Code Online (Sandbox Code Playgroud)

  • API 28 开始有一个新的 api 来获取进程名称,性能更高。查看 https://developer.android.com/reference/android/app/Application.html#getProcessName() (5认同)
  • 这是一个小的修改:if(Build.VERSION.SDK_INT> = Build.VERSION_CODES.P){字符串处理= getProcessName(this); 如果(!getPackageName()。equals(process))WebView.setDataDirectorySuffix(process); } (3认同)

Mic*_*cer 5

为了总结所有改进的修复,这是 Kotlin 中的代码:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
    if (packageName != Application.getProcessName()) {
        WebView.setDataDirectorySuffix(Application.getProcessName())
    }
}
Run Code Online (Sandbox Code Playgroud)

将其添加到您的ApplicationonCreate()方法中。

请注意,这只能解决最多 2 个进程的问题。如果您的应用程序使用更多,则必须为每个应用程序提供不同的 WebView 后缀。