为啥 context.startActivity(intent) 没有启动活动以及如何在 android 中处理异常?

Posted

技术标签:

【中文标题】为啥 context.startActivity(intent) 没有启动活动以及如何在 android 中处理异常?【英文标题】:Why context.startActivity(intent) not starting the activity and how to handle exception in android?为什么 context.startActivity(intent) 没有启动活动以及如何在 android 中处理异常? 【发布时间】:2020-11-30 12:05:43 【问题描述】:

我有一个异常处理程序来处理来自 Activity 类的异常,异常处理程序如下所示。

public class ExceptionHandler implements Thread.UncaughtExceptionHandler 
    public static final String TAG = "Exception handler";
    private final Context activity;

    public ExceptionHandler(Context activity) 
        this.activity = activity;
    

    @Override
    public void uncaughtException(@NonNull Thread thread, @NonNull Throwable throwable) 
        Intent error = new Intent(activity, ErrorCatcher.class);
        activity.startActivity(error);
    

从活动类初始化

  @Override
    protected void onCreate(Bundle savedInstanceState) 
        Log.e(TAG, "onRestart: Hey just created");
        Thread.setDefaultUncaughtExceptionHandler(new ExceptionHandler(this.getApplicationContext()));
// other methods and function

当控件来到异常处理程序时,并没有创建活动,而是挂起我的应用程序的空白页面。

我得到的唯一消息是

I/Timeline: Timeline: Activity_launch_request time:417281208 intent:Intent  cmp=com.yyy.xxx/com.yyy.xxx.Activities.Error 

编辑:好的,经过一番挖掘,我发现,如果没有抛出异常,那么活动就会启动(即我可以看到活动页面)但是当抛出异常时,就会出现空白页面被展示。为什么只有在抛出异常时才会出现这种情况,以及在android中处理异常的更好方法是什么?有什么帮助吗?


EDIT 2:类似于https://codecrunch.co/blog/custom-error-handling-in-android/

【问题讨论】:

我会重命名您的目标活动,因为 Error 可能会与 java.lang.Error 混淆,尤其是当您在同一上下文中处理 java.lang.Throwable 时。 但这不可能是它无法正常工作的原因?? 【参考方案1】:

在 startActivity 之前,添加error.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK);。还有更完整的答案here

【讨论】:

这些是一些很棒的答案,谢谢,一旦这些答案解决了我的问题,我会接受你的答案。 没关系,它根本不工作,显示与以前相同的状态...... 尝试在 Application 构造函数中设置异常处理程序,而不是在 Activity onCreate 中设置。添加一个将 Application 扩展至基本文件夹的类,在其构造函数中设置线程 uncaughtexceptionhandler。您可以传递“this”而不是“this.getApplicationContext()”并查看它是否加载了您的活动。 我创建了一个project on github,尝试在有和没有“抛出异常”的情况下编译它,看看有什么不同。 记得在 android manifest 中添加应用程序类以使其工作,有时我会忘记这样做。【参考方案2】:

此代码可能对您有所帮助

Application application = (Application) context.getApplicationContext();
Intent intent = new Intent(application, ErrorActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
application.startActivity(intent);

如果没有,请通知我,我会发给你完整的异常处理代码!

【讨论】:

嘿,似乎可行,但有一个问题,作为 leaonardo 提供的链接,我面临与链接中的 OP 相同的问题,我在活动开始之前不断收到崩溃消息...可以你帮我解决这个问题? 没关系,它根本不工作,显示与以前相同的状态......【参考方案3】:

当我使用 try/catch 代替时它起作用了

Thread.setDefaultUncaughtExceptionHandler(new ExceptionHandler(this.getApplicationContext()));

要处理异常,它可能不是最好的方法,但值得一试。

【讨论】:

【参考方案4】:

首先,您正在使用Application Context 尝试启动一项活动。这可能会导致您的一些问题。这是可以做到的,但我认为有一种更简单的方法可以解决所有这些问题。

还应该注意,你忘记了里面的其余代码

  @Override
public void uncaughtException
    ....
    android.os.Process.killProcess(android.os.Process.myPid());
    System.exit(10);

为什么只有在抛出异常时才会出现这种情况...

这是因为你重写了uncaughtException 方法。所以当你得到一个未捕获的异常时,你将启动Error.class Activity。

  @Override
    public void uncaughtException(@NonNull Thread thread, @NonNull Throwable throwable) 
        Intent error = new Intent(activity, ErrorCatcher.class);
        activity.startActivity(error);
    

...在android中处理异常的更好方法是什么?

您可以通过多种方式处理异常。这实际上取决于用例是什么。一般来说,您应该在错误发生时处理它们,至少在开始时是这样。您可以在可能引发错误的方法上使用 try/catch 块来执行此操作;或者,您可以通过创建引发错误的方法来“冒泡错误”以在适当的位置处理它,具体取决于您的应用程序的架构。如果没有更多细节,我无法就处理错误的最佳方式提供准确的响应。在我的应用程序中,如果需要用户通知,我通常会举杯祝酒,如果出现网络故障等问题,我通常会举杯。

异常处理是一个复杂的问题,应该这样处理。对于如何处理错误,没有单一的正确答案。但是,一些基础知识是:

    处理所有异常。如果您有一个引发错误的方法,请确保您包含在 try/catch 中并进行适当处理。这应该可以防止任何“未捕获的异常”。

    具体说明您正在处理的错误。您似乎正在尝试为您的活动中的任何未捕获错误创建一个“包罗万象”。我不会那样做的。我会将所有代码包装在try/catch 中,并在出现错误时对其进行处理。当我想到你的代码时,我猜你得到一个空白屏幕的原因是你没有为你的错误类创建一个布局,但我可能是错的。

TLDR;如果我是你,我会摆脱:

@Override
public void uncaughtException(@NonNull Thread thread, @NonNull Throwable throwable) 
    Intent error = new Intent(activity, ErrorCatcher.class);
    activity.startActivity(error);

也摆脱这个:

protected void onCreate(Bundle savedInstanceState) 
    Log.e(TAG, "onRestart: Hey just created");
    //Thread.setDefaultUncaughtExceptionHandler(new ExceptionHandler(this.getApplicationContext()));

从那里我会找出哪些方法会引发错误并在您调用它们时处理它们。这将是最简单的方法。从那里您可以考虑一种更大、更复杂的错误处理方法。您不一定需要为错误创建新活动。如果你愿意,我会在 finally 块中进行。确保您已经为错误活动创建了布局;否则,它将是空白的;将错误活动添加到清单中(否则会崩溃)。

try
   //your add all your method calls here.
 catch (Exception E)
   //handle generic exception
 catch (ThreadException e)
   //handle the thread exception
   ...add more catches for more specific errors
 finally
   //Optional call once all exceptions have been caught
   Intent error = new Intent(this, ErrorCatcher.class);
   startActivity(error);

【讨论】:

捕获所有未捕获的异常是有原因的,所以我可以稍后再捕获它们,而且如果发生异常,除了重新启动应用程序(至少基于该应用程序)虽然我可能是错的,但布局也存在并且还填充了设计。最后我有理由不摆脱 uncaughtException ,但如果你能告诉我为什么会发生这种情况,那将是很大的帮助。 我在回答的第一段中告诉过你为什么会发生这种情况。 归根结底,我从未见过有人像你这样处理错误,而且我已经开发企业级代码近十年了。 我更新了我的答案,因为您似乎遗漏了一些可能有用的代码。祝你好运。 我认为问这个问题的主要原因是捕捉所有异常,以便您的应用在没有反馈的情况下不会崩溃,而不是改进错误纠正。无法预测您的应用程序在负载不同的设备中会发生什么,并且随着复杂性的增加,情况会变得更糟。在我看来,最好的错误纠正方法是使用像 Crashlytics 这样的日志服务,当你需要尝试/捕获一些异常时,你会在异常发生的地方这样做,并使用这个自定义异常处理程序作为安全网以获得更好的用户体验.就个人而言,我只是让它崩溃。【参考方案5】:

在您覆盖的 uncaughtException 方法上,您最好使用此代码启动活动

   startActivity(activity,Error.class);

【讨论】:

startActivity 不能被非活动类直接调用,即不扩展 Activity 类的类。如果我使用context.startActivity 或只是startActivity 这将无济于事,因为两者仍将调用相同的函数

以上是关于为啥 context.startActivity(intent) 没有启动活动以及如何在 android 中处理异常?的主要内容,如果未能解决你的问题,请参考以下文章

Context.startActivity() 与 Activity.startActivity() 究竟有什么不同?

Context.startActivity() 与 Activity.startActivity() 究竟有什么不同?

startActivity kotlin 的一种写法

startActivity kotlin 的一种写法

浅显易懂的startActivity原理分析

Activity的启动过程详细分析