- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在设计一个简单的玩具解释器,并且我有一个自定义异常,如下所示:
class ValError(varName: String) : Exception("$varName is val, cannot reassign.")
问题是,当它被抛出时,它会打印出整个调用堆栈跟踪,如下所示:
ValError: foo is val, cannot reassign.
at com.github.me.dynamik.interpreter.Environment.get(Environment.kt:62)
at com.github.me.dynamik.interpreter.TreeWalker.visitVariableExpr(TreeWalker.kt:154)
at com.github.me.dynamik.expressions.VariableExpr.evaluateBy(Expression.kt:57)
但我只想打印出第一行
ValError: foo is val, cannot reassign.
我该如何实现这个目标?
最佳答案
简短回答:捕获异常,并打印其 message字段。您可以在顶级函数的 try
...catch
block 中执行此操作,或者(更好)在 UncaughtExceptionHandler 中执行此操作。 .
冗长无聊的答案:
异常将从每个函数向上传递到其调用者,直到找到带有封闭的 try
…catch
block 的异常。
因此一种方法是自己捕获异常,例如:
fun main() {
try {
// All your top-level code here…
} catch (x: Exception) {
System.err.println(x.message)
}
}
如果您正在使用日志记录框架,当然可以使用它。
如果没有封闭的try
...catch
,它将调用线程的UncaughtExceptionHandler — 或者,如果该线程没有,则其 ThreadGroup是其中之一,或者如果失败的话,the default one .
如果您不提供默认值,系统将调用其 printStackTrace()方法,它将异常消息及其所有回溯打印到标准错误流 - 这就是您看到问题中显示的跟踪的原因。
因此另一种方法是设置 UncaughtExceptionHandler在这些级别之一,例如:
Thread.setDefaultUncaughtExceptionHandler { t, x ->
System.err.println(x.message)
}
这更安全,因为它适用于所有线程。
如果您正在编写一个仅使用单个线程的简单应用程序,那么在顶级方法中捕获异常可以很简单。但是,如果您正在编写使用 GUI 工具包或 Web 框架,或者使用协程或其他执行框架,或者手动启动任何线程的内容,那么可能还有其他线程在运行。这有两个效果:
首先,您将无权访问这些线程的顶级方法,因此您将无法捕获其中的异常。所以UncaughtExceptionHandler是唯一的办法。 (默认的是最安全的,因为您可能无法访问线程,甚至无法访问所涉及的线程组。)
其次,虽然出现异常的线程将停止运行,但任何其他线程都不会。因此 JVM 不会关闭,其他线程将继续运行,而不会察觉。这会使您的应用程序处于不一致的状态,这可能是灾难性的:您最终可能会遇到作业排队但从未运行,或者其他线程停滞等待数据,或者请求未得到答复,或者几乎任何事情,具体取决于您的应用程序的情况结构化以及哪个线程死亡。而且由于 JVM 仍在继续,系统级监视器或服务管理器不会知道出现任何问题。
因此,在记录异常后,在某些情况下,处理程序关闭整个应用程序可能是一个非常好的主意。
(顺便说一句,第三种方法是使用自定义 Exception
并重写其 printStackTrace()
方法以仅打印消息。但这不是一个好主意;它违背了该方法的意图,这可能会给使用它的任何东西带来各种各样的问题。)
关于kotlin - 如何只显示错误而不显示调用堆栈跟踪?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55883750/
我是一名优秀的程序员,十分优秀!