jjzjj

android - Firebase 远程配置 : NullPointerException on Map. keySet()

coder 2023-12-13 原文

很多用户遇到以下崩溃问题:

Fatal Exception: java.lang.NullPointerException: Attempt to invoke interface method 'java.util.Set java.util.Map.keySet()' on a null object reference
   at com.google.android.gms.internal.zzbtn.zza(Unknown Source)
   at com.google.android.gms.internal.zzbtn.run(Unknown Source)
   at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
   at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
   at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
   at java.lang.Thread.run(Thread.java:818)

我搜索了外部库,发现崩溃的来源是以下类 zzbtn 中的 firebase-config 库(参见方法 - zza())。

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//

package com.google.android.gms.internal;

import android.content.Context;
import android.util.Log;
import com.google.android.gms.internal.zzbtl;
import com.google.android.gms.internal.zzbto;
import com.google.android.gms.internal.zzbtr;
import com.google.android.gms.internal.zzbxt;
import com.google.android.gms.internal.zzbts.zza;
import com.google.android.gms.internal.zzbts.zzb;
import com.google.android.gms.internal.zzbts.zzc;
import com.google.android.gms.internal.zzbts.zzd;
import com.google.android.gms.internal.zzbts.zze;
import com.google.android.gms.internal.zzbts.zzf;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

public class zzbtn implements Runnable {
public final Context mContext;
public final zzbto zzclY;
public final zzbto zzclZ;
public final zzbto zzcma;
public final zzbtr zzclQ;

    public zzbtn(Context var1, zzbto var2, zzbto var3, zzbto var4, zzbtr var5) {
        this.mContext = var1;
        this.zzclY = var2;
        this.zzclZ = var3;
        this.zzcma = var4;
        this.zzclQ = var5;
    }

    private zza zza(zzbto var1) {
        zza var2 = new zza();
        if(var1.zzace() != null) {
            Map var3 = var1.zzace();
            ArrayList var4 = new ArrayList();
            Iterator var5 = var3.keySet().iterator();

            while(var5.hasNext()) {
                String var6 = (String)var5.next();
                ArrayList var7 = new ArrayList();
                Map var8 = (Map)var3.get(var6);
                /* Crash is here when reading keySet() on null map */
                Iterator var9 = var8.keySet().iterator();

                while(var9.hasNext()) {
                    String var10 = (String)var9.next();
                    zzb var11 = new zzb();
                    var11.zzaB = var10;
                    var11.zzcml = (byte[])var8.get(var10);
                    var7.add(var11);
                }

                zzd var16 = new zzd();
                var16.zzaGP = var6;
                zzb[] var12 = new zzb[var7.size()];
                var16.zzcmq = (zzb[])var7.toArray(var12);
                var4.add(var16);
            }

            zzd[] var15 = new zzd[var4.size()];
            var2.zzcmi = (zzd[])var4.toArray(var15);
        }

        if(var1.zzzD() != null) {
            List var13 = var1.zzzD();
            byte[][] var14 = new byte[var13.size()][];
            var2.zzcmj = (byte[][])var13.toArray(var14);
        }

        var2.timestamp = var1.getTimestamp();
        return var2;
    }

    public void run() {
        zze var1 = new zze();
        if(this.zzclY != null) {
            var1.zzcmr = this.zza(this.zzclY);
        }

        if(this.zzclZ != null) {
            var1.zzcms = this.zza(this.zzclZ);
        }

        if(this.zzcma != null) {
            var1.zzcmt = this.zza(this.zzcma);
        }

        if(this.zzclQ != null) {
            zzc var2 = new zzc();
            var2.zzcmm = this.zzclQ.getLastFetchStatus();
            var2.zzcmn = this.zzclQ.isDeveloperModeEnabled();
            var2.zzcmo = this.zzclQ.zzacj();
            var1.zzcmu = var2;
        }

        if(this.zzclQ != null && this.zzclQ.zzach() != null) {
            ArrayList var8 = new ArrayList();
            Map var3 = this.zzclQ.zzach();
            Iterator var4 = var3.keySet().iterator();

            while(var4.hasNext()) {
                String var5 = (String)var4.next();
                if(var3.get(var5) != null) {
                    zzf var6 = new zzf();
                    var6.zzaGP = var5;
                    var6.zzcmx = ((zzbtl)var3.get(var5)).zzacd();
                    var6.resourceId = ((zzbtl)var3.get(var5)).zzacc();
                    var8.add(var6);
                }
            }

            zzf[] var11 = new zzf[var8.size()];
            var1.zzcmv = (zzf[])var8.toArray(var11);
        }

        byte[] var9 = zzbxt.zzf(var1);

        try {
            FileOutputStream var10 = this.mContext.openFileOutput("persisted_config", 0);
            var10.write(var9);
            var10.close();
        } catch (IOException var7) {
            Log.e("AsyncPersisterTask", "Could not persist config.", var7);
        }

    }
}

此类在 FirebaseRemoteConfig 类中使用(参见方法 zzt()),如下所示:

https://gist.github.com/anonymous/e6f23c1dc37bf905a9224d8b72ab6cd9

我在我的应用程序类中使用 FirebaseRemoteConfig 如下:

public class MyApp extends MultiDexApplication {

    public static boolean sound = true;
    private static Context context;

    private Typeface regularTypeFace;
    private Typeface boldTypeFace;
    private final String LOG_TAG = "MyApp";
    private FirebaseRemoteConfig remoteConfig = null;

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

        Fabric.with(this, new Crashlytics());

        context = getApplicationContext();

        /* Try catch to handle any runtime exception thrown by firebase API */
        try {
            /* Initialize firebase app : Done to avoid crashes due to IllegalStateException - Default FirebaseApp is not initialized */
            FirebaseApp.initializeApp(this);
            /* Start to fetch Remote config parameters */
            startConfigFetch();
        } catch (Exception e){
            e.printStackTrace();
        }

    }

    /**
     * Description : Fetches remote config params from firebase & uses the fetched values
     */
    public void startConfigFetch() {

        /* try to get the default instance of Firebase Remote config */
        /* try-catch : To Resolve Crash #1399 */
        try {
            remoteConfig = FirebaseRemoteConfig.getInstance();
        } catch (IllegalStateException e){
            e.printStackTrace();
        }

        /* If we don't get an instance of Firebase remote config, then do nothing */
        if (remoteConfig == null){
            return;
        }

        FirebaseRemoteConfigSettings remoteConfigSettings = new FirebaseRemoteConfigSettings.Builder()
                .setDeveloperModeEnabled(BuildConfig.DEBUG)
                .build();

        remoteConfig.setConfigSettings(remoteConfigSettings);
        remoteConfig.setDefaults(R.xml.remote_config_defaults);

        /* Time for which cache lives, for now its 0 ms */
        long cacheExpiration = 0;

        OnCompleteListener<Void> onCompleteListener = new OnCompleteListener<Void>() {
            @Override
            public void onComplete(@NonNull Task<Void> task) {

                if (task.isSuccessful()) {
                    if (getContext() != null) {
                        onFetchConfigSuccess();
                    }
                } else {
                    Log.d(LOG_TAG, "Stories Fetch Fail");
                }
            }
        };

        OnFailureListener onFailListener = new OnFailureListener() {
            @Override
            public void onFailure(@NonNull Exception e) {
                Log.d(LOG_TAG, "Stories Fetch Fail in remote config");
            }
        };

        remoteConfig.fetch(cacheExpiration).addOnCompleteListener(onCompleteListener).addOnFailureListener(onFailListener);
    }

    /**
     * Called when fetch config is success
     * */
    private void onFetchConfigSuccess() {

        /* Once the config is successfully fetched it must be activated before newly fetched */
        /* values are returned (or can be used) */
        remoteConfig.activateFetched();

        /* Get dynamic stories string */
        String dynamicStories = remoteConfig.getString(Common.KEY_DYNAMIC_STORIES_FIREBASE_CONFIG);
        /* Get bundled stories (these are app-bundled stories & dynamic stories but part of bundled stories) */
        String bundledStories = remoteConfig.getString(Common.KEY_BUNDLED_STORIES_FIREBASE_CONFIG);

    /* Do something with dynamicStories & bundledStories values */
    }

}

类 zzbto 在这里:

https://gist.github.com/anonymous/e2f3a67e6fd3be51ba4456fe2e847890

请帮助解决此崩溃。

最佳答案

Kirril Karmizin 中所述的评论,这是自 2017 年 6 月以来的已知错误 (Github Firebase issues)。我还收到了多个崩溃报告

Fatal Exception: java.lang.NullPointerException
   at com.google.android.gms.internal.zzdyx.zzb(Unknown Source)
   at com.google.android.gms.internal.zzdyx.zza(Unknown Source)
   at com.google.android.gms.internal.zzfan.run(Unknown Source)
   at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:231)
   at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
   at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
   at java.lang.Thread.run(Thread.java:848)

,有:

  • 火力地堡:12.0.1
  • 构建工具:27.0.3

对于那些希望在其应用程序中保留 Remote Config 的人来说,在正确方向上可能的一步可以在 Official RemoteConfic docs 中找到。 :

You should set in-app default parameter values in the Remote Config object, so that your app behaves predictably before it fetches values from the Remote Config service.

对我来说,我不能冒在生产应用中出现更多崩溃报告的风险。我删除了 Remote Config,直到现在我还没有看到崩溃。

关于android - Firebase 远程配置 : NullPointerException on Map. keySet(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42362024/

有关android - Firebase 远程配置 : NullPointerException on Map. keySet()的更多相关文章

  1. ruby-on-rails - 独立 ruby​​ 脚本的配置文件 - 2

    我有一个在Linux服务器上运行的ruby​​脚本。它不使用rails或任何东西。它基本上是一个命令行ruby​​脚本,可以像这样传递参数:./ruby_script.rbarg1arg2如何将参数抽象到配置文件(例如yaml文件或其他文件)中?您能否举例说明如何做到这一点?提前谢谢你。 最佳答案 首先,您可以运行一个写入YAML配置文件的独立脚本:require"yaml"File.write("path_to_yaml_file",[arg1,arg2].to_yaml)然后,在您的应用中阅读它:require"yaml"arg

  2. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  3. Vscode+Cmake配置并运行opencv环境(Windows和Ubuntu大同小异) - 2

    之前在培训新生的时候,windows环境下配置opencv环境一直教的都是网上主流的vsstudio配置属性表,但是这个似乎对新生来说难度略高(虽然个人觉得完全是他们自己的问题),加之暑假之后对cmake实在是爱不释手,且这样配置确实十分简单(其实都不需要配置),故斗胆妄言vscode下配置CV之法。其实极为简单,图比较多所以很长。如果你看此文还配不好,你应该思考一下是不是自己的问题。闲话少说,直接开始。0.CMkae简介有的人到大二了都不知道cmake是什么,我不说是谁。CMake是一个开源免费并且跨平台的构建工具,可以用简单的语句来描述所有平台的编译过程。它能够根据当前所在平台输出对应的m

  4. 神州数码无线产品(AC+AP)配置 - 2

    注意:本文主要掌握DCN自研无线产品的基本配置方法和注意事项,能够进行一般的项目实施、调试与运维AP基本配置命令AP登录用户名和密码均为:adminAP默认IP地址为:192.168.1.10AP默认情况下DHCP开启AP静态地址配置:setmanagementstatic-ip192.168.10.1AP开启/关闭DHCP功能:setmanagementdhcp-statusup/downAP设置默认网关:setstatic-ip-routegeteway192.168.10.254查看AP基本信息:getsystemgetmanagementgetmanaged-apgetrouteAP配

  5. hadoop安装之保姆级教程(二)之YARN的配置 - 2

    1.1.1 YARN的介绍 为克服Hadoop1.0中HDFS和MapReduce存在的各种问题⽽提出的,针对Hadoop1.0中的MapReduce在扩展性和多框架⽀持⽅⾯的不⾜,提出了全新的资源管理框架YARN. ApacheYARN(YetanotherResourceNegotiator的缩写)是Hadoop集群的资源管理系统,负责为计算程序提供服务器计算资源,相当于⼀个分布式的操作系统平台,⽽MapReduce等计算程序则相当于运⾏于操作系统之上的应⽤程序。 YARN被引⼊Hadoop2,最初是为了改善MapReduce的实现,但是因为具有⾜够的通⽤性,同样可以⽀持其他的分布式计算模

  6. 安卓apk修改(Android反编译apk) - 2

    最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路

  7. Ruby 默认将 IRB 配置为 Pretty_Inspect - 2

    我是ruby​​的新手,正在配置IRB。我喜欢pretty-print(需要'pp'),但总是输入pp来漂亮地打印它似乎很麻烦。我想做的是默认情况下让它漂亮地打印出来,所以如果我有一个var,比如说,'myvar',然后键入myvar,它会自动调用pretty_inspect而不是常规检查。我从哪里开始?理想情况下,我将能够向我的.irbrc文件添加一个自动调用的方法。有什么想法吗?谢谢! 最佳答案 irb中默认pretty-print对象正是hirb被迫去做。Theseposts解释hirb如何将几乎所有内容转换为ascii表。虽

  8. ruby - 是否可以将 IRB 提示配置为动态更改? - 2

    我想在IRB中浏览文件系统并让提示更改以反射(reflect)当前工作目录,但我不知道如何在每个命令后进行提示更新。最终,我想在日常工作中更多地使用IRB,让bash溜走。我在我的.irbrc中试过这个:require'fileutils'includeFileUtilsIRB.conf[:PROMPT][:CUSTOM]={:PROMPT_N=>"\e[1m:\e[m",:PROMPT_I=>"\e[1m#{pwd}>\e[m",:PROMPT_S=>"FOO",:PROMPT_C=>"\e[1m#{pwd}>\e[m",:RETURN=>""}IRB.conf[:PROMPT_MO

  9. ruby - 如何配置 Ruby Mechanize 代理以通过 Charles Web 代理工作? - 2

    我正在使用Ruby/Mechanize编写一个“自动填写表格”应用程序。它几乎可以工作。我可以使用精彩CharlesWeb代理以查看服务器和我的Firefox浏览器之间的交换。现在我想使用Charles查看服务器和我的应用程序之间的交换。Charles在端口8888上代理。假设服务器位于https://my.host.com。.一件不起作用的事情是:@agent||=Mechanize.newdo|agent|agent.set_proxy("my.host.com",8888)end这会导致Net::HTTP::Persistent::Error:...lib/net/http/pe

  10. ruby-on-rails - 如果特定语言环境中缺少翻译,如何配置 i18n 以使用 en 语言环境? - 2

    如果特定语言环境中缺少翻译,如何配置i18n以使用en语言环境翻译?当前已插入翻译缺失消息。我正在使用RoR3.1。 最佳答案 找到相似的question这里是答案:#application.rb#railswillfallbacktoconfig.i18n.default_localetranslationconfig.i18n.fallbacks=true#railswillfallbacktoen,nomatterwhatissetasconfig.i18n.default_localeconfig.i18n.fallback

随机推荐