有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java为什么Clojure用未检查的异常包装检查的异常?

下面的代码演示了这样一种情况:一个操作抛出一个选中的异常ExecutionException,Clojure用RuntimeException将其包装

Clojure为什么要这么做?这正常吗?Clojure似乎在做一些与Java不同的事情。在本例中,处理导致失败的实际异常Exception的惯用方法是什么

user=> (def f (future (Thread/sleep 10000) (throw (Exception. "hello world"))))
#'user/f

user=> (.get f)
Exception hello world  user/fn--318 (NO_SOURCE_FILE:81)

user=> (.printStackTrace *e)
java.lang.RuntimeException: java.util.concurrent.ExecutionException: java.lang.Exception: hello world
    at clojure.lang.Util.runtimeException(Util.java:165)
    at clojure.lang.Reflector.invokeMatchingMethod(Reflector.java:97)
    at clojure.lang.Reflector.invokeNoArgInstanceMember(Reflector.java:316)
    at user$eval320.invoke(NO_SOURCE_FILE:82)
    at clojure.lang.Compiler.eval(Compiler.java:6465)
    at clojure.lang.Compiler.eval(Compiler.java:6431)
    at clojure.core$eval.invoke(core.clj:2795)
    at clojure.main$repl$read_eval_print__5967.invoke(main.clj:244)
    at clojure.main$repl$fn__5972.invoke(main.clj:265)
    at clojure.main$repl.doInvoke(main.clj:265)
    at clojure.lang.RestFn.invoke(RestFn.java:421)
    at clojure.main$repl_opt.invoke(main.clj:331)
    at clojure.main$main.doInvoke(main.clj:427)
    at clojure.lang.RestFn.invoke(RestFn.java:397)
    at clojure.lang.Var.invoke(Var.java:397)
    at clojure.lang.AFn.applyToHelper(AFn.java:159)
    at clojure.lang.Var.applyTo(Var.java:518)
    at clojure.main.main(main.java:37)
Caused by: java.util.concurrent.ExecutionException: java.lang.Exception: hello world
    at java.util.concurrent.FutureTask$Sync.innerGet(FutureTask.java:222)
    at java.util.concurrent.FutureTask.get(FutureTask.java:83)
    at clojure.core$future_call$reify__5684.get(core.clj:6064)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at clojure.lang.Reflector.invokeMatchingMethod(Reflector.java:92)
    ... 16 more
Caused by: java.lang.Exception: hello world
    at user$fn__318.invoke(NO_SOURCE_FILE:81)
nil
    at clojure.core$binding_conveyor_fn$fn__3713.invoke(core.clj:1817)
    at clojure.lang.AFn.call(AFn.java:18)
    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
    at java.util.concurrent.FutureTask.run(FutureTask.java:138)
    at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
    at java.lang.Thread.run(Thread.java:680)

共 (2) 个答案

  1. # 1 楼答案

    请注意,ExceptionException确实被抛出——只是异常被Clojure捕获并处理

    Clojure在处理某些异常时会将它们包装在RuntimeException中——我相信这样做是为了避免在Clojure源代码中处理检查过的异常

    如果您想访问底层异常,那么(.getCause e)应该可以工作

  2. # 2 楼答案

    根据这个链接,clojure似乎不会抛出选中的异常,所以我假设clojure捕获任何可能抛出的异常并将其包装: http://dev.clojure.org/display/doc/1.3

    至于惯用的处理方式(?)在Java中,旧规则仍然适用。我会检查ExecutionException中包装的异常是否是我预期可能发生的已检查异常之一,并像在任何其他情况下一样处理它们(阅读:模仿编译器强制执行的try-catch)。如果不是,我会将其包装在运行时异常中并再次抛出,或者根据情况记录并吞下它