Android app会crash的原因及解决方法

 更新时间:2020年12月29日 10:30:32   作者:Android开发中文站  
这篇文章主要介绍了Android app会crash的原因及解决方法,帮助大家更好的进行Android开发,感兴趣的朋友可以了解下

android main入口的commonInit()方法内处,有这么一句话,

Thread.setDefaultUncaughtExceptionHandler(new KillApplicationHandler(loggingHandler));

如果没有这句话,app就不会crash。不信,你往里面看,

public KillApplicationHandler(LoggingHandler loggingHandler) {
 @Override
 public void uncaughtException(Thread t, Throwable e) {
 //捕获到异常
   try {
     ......
     //打印crash日志,展示崩溃弹窗等
     // Bring up crash dialog, wait for it to be dismissed
     ActivityManager.getService().handleApplicationCrash(
       mApplicationObject, new ApplicationErrorReport.ParcelableCrashInfo(e));
      } catch (Throwable t2) {
        .... 
      } finally {
        // Try everything to make sure this process goes away.
        Process.killProcess(Process.myPid());//杀死进程
        System.exit(10);
      }
    }
  } 

当异常KillApplicationHandler捕获到异常,进行完一系列处理(主要是打印crash日志,通知AMS展示crash弹窗等)后,最终会杀死进程,这样你的app就崩溃了。

既然都崩溃了,自定义异常捕获器来屏蔽crash真的可行吗?

肯定有人会说,自定义一个异常捕获器,来覆盖掉系统的KillApplicationHandler,然后在捕获到异常后,不杀进程,app就不会崩溃了,就像下面这样,

class MainApplication : Application() {
  override fun onCreate() {
    super.onCreate()
    Thread.setDefaultUncaughtExceptionHandler { _, e ->
      //捕获到异常,只打印日志,不杀进程
      Log.e("MainApplication", "${Thread.currentThread().name} 捕获到异常:${e.message}")
    }
  }
}

这其实只是隔壁老王的思路,虽然确实防护住子线程的crash,但是当主线程出现异常时,app还是无法正常运行。这是因为,当UncaughtExceptionHandler捕获到线程抛出异常的时候,线程在执行完uncaughtException()中的处理后,就无法继续存活了。如果抛异常的线程是主线程,那就意味着主线程会死掉,这时你即便不杀进程,进程活着也没有任何意义了,app还是会停止运行。

把android异常捕获机制在梳理一下,熟悉的同学可以跳过,直接进入下一节。

  • Thread.setCaughtExceptionPreHandler()覆盖所有线程,会在回调DefaultExceptionHandler之前调用;
  • Thread.setCaughtExceptionHandler()同样回覆盖所有线程,可以在应用层被重复调用,并且每一次调用后,都会覆盖上一次设置的DefaultUncaughtExceptionHandler;
  • Thread.currentThread.setUncaughtExceptionHandler(),只可以覆盖当前线程的异常。如果某个线程存在自定义的UncaughtExceptionHandler,回调时会忽略全局的DefaultUncaughtHandler。

既然话都说到这份上了,就请接下never crash大招吧。

要想不crash,只能让线程不要抛出exception,唯此别无他法。如果我们能把一个线程的所有的操作都使用try-catch进行保护,理论上,就能做到app never crash。由于android基于Handler事件驱动的机制,可以在app启动时,向主线程中的MessageQueue中提交一个死循环操作,在这个死循环中不断去poll事件,并且将这个死循环进行try-catch,这样所有主线程中的异常都会被catch住,从而app就再也不会发生crash。

private fun openCrashProtected() {
    Log.d(tag, "openCrashProtected")
    Handler(Looper.getMainLooper()).post {
      while (true) {
        try {
          Looper.loop()
          Log.d(tag, "main looper execute loop")
        } catch (e: Throwable) {
          //所有主线程中的异常都会被catch住,从而不会发生crash
          Log.e(tag, "catch exception: " + e.message)
        }
      }
    }
  }

有人可能要说了,你这样catch住主线程的异常了,页面可能要乱套哇。话虽如此,但你可以在catch中做业务保护呀。比如,我这里采取的做法是,关闭栈顶activity。 解决ActivityLifeCycle,维护一个Activity栈,

private fun registerLifeCycle() {
    registerActivityLifecycleCallbacks(object : ActivityLifecycleCallbacks {
      override fun onActivityCreated(activity: Activity, savedInstanceState: Bundle?) {
        ActivityStack.Instance().push(activity)
      }

      override fun onActivityResumed(activity: Activity) {
      }

      override fun onActivityStarted(activity: Activity) {
      }

      override fun onActivityPaused(activity: Activity) {

      }

      override fun onActivityDestroyed(activity: Activity) {
        ActivityStack.Instance().pop(activity)
      }

      override fun onActivitySaveInstanceState(activity: Activity, outState: Bundle) {

      }

      override fun onActivityStopped(activity: Activity) {
      }
    })
  }

然后当catch住异常时,

//主线程出现异常,关闭栈顶activity
ActivityStack.Instance().curr()?.finish()

github代码

最后奉上github仓库代码,请笑纳。

以上就是Android app会crash的原因及解决方法的详细内容,更多关于Android app crash的资料请关注脚本之家其它相关文章!

相关文章

  • 解析Android框架之Volley源码

    解析Android框架之Volley源码

    我们知道Volley是在2013年Google I/O大会上推出了一个新的网络通信框架,他的设计目的就是为了那些请求数据量不是特别大,但是又是特别频繁的网络操作非常适合。但是对于数据量较大的请求,比如说下载一个较大的文件,Volley可能相比于其他的框架,就有点不足了。
    2021-06-06
  • Android4.0平板开发之隐藏底部任务栏的方法

    Android4.0平板开发之隐藏底部任务栏的方法

    这篇文章主要介绍了Android4.0平板开发之隐藏底部任务栏的方法,结合实例形式较为详细的分析了Android隐藏于显示底部任务栏的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-11-11
  • Android编程实现画板功能的方法总结【附源码下载】

    Android编程实现画板功能的方法总结【附源码下载】

    这篇文章主要介绍了Android编程实现画板功能的方法,结合实例形式总结分析了Android基于自定义View与Canvas类实现画板功能的具体操作步骤与相关注意事项,需要的朋友可以参考下
    2018-02-02
  • myeclipse android环境搭建图文教程

    myeclipse android环境搭建图文教程

    在从事android应用开发过程中,搭配环境是所有开始的第一步,本文将介绍myeclipse android环境搭建图文教程,希望可以帮助大家完成android开发之旅
    2012-11-11
  • Android微信第三方登录(个人笔记)

    Android微信第三方登录(个人笔记)

    这篇文章主要为大家详细介绍了Android微信第三方登录的具体过程,个人笔记分享,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • Android开发之在xml中设置自定义属性的方法

    Android开发之在xml中设置自定义属性的方法

    下面小编就为大家分享一篇Android开发之在xml中设置自定义属性的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01
  • Flutter fluro时报错type 'String' is not a subtype of type 'Queue<Task>'

    Flutter fluro时报错type 'String' is not a subty

    这篇文章主要介绍了Flutter使用fluro时报错type 'String' is not a subtype of type 'Queue<Task>'解决方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2023-12-12
  • Android strings.xml使用方法及技巧

    Android strings.xml使用方法及技巧

    本文主要介绍Android strings.xml,这里对strings.xml文件的使用和一些小技巧做了总结,有兴趣的同学可以看下
    2016-08-08
  • Android 仿微信小程序入口动画

    Android 仿微信小程序入口动画

    突然发现微信下拉小程序入口动画非常细腻,比较好奇,所以仿照他做了一个,并不是很完美,部分效果还没完成,但总体自我感觉还不错,效果见下文
    2021-06-06
  • Android 实时监测(监听)网络连接状态变化

    Android 实时监测(监听)网络连接状态变化

    这篇文章主要介绍了Android 实时监测(监听)网络连接状态变化的相关知识,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-06-06

最新评论