誰かがこのコードを説明できますか?
public class SneakyThrow {
public static void sneakyThrow(Throwable ex) {
SneakyThrow.<RuntimeException>sneakyThrowInner(ex);
}
private static <T extends Throwable> T sneakyThrowInner(Throwable ex) throws T {
throw (T) ex;
}
public static void main(String[] args) {
SneakyThrow.sneakyThrow(new Exception());
}
}
奇妙に思えるかもしれませんが、これはキャスト例外を生成せず、署名で宣言することなくチェック例外をスローしたり、チェックされていない例外にラップしたりすることを許可します。
どちらもsneakyThrow(...)
メインもチェック例外を宣言していませんが、出力は次のようになります。
Exception in thread "main" java.lang.Exception
at com.xxx.SneakyThrow.main(SneakyThrow.java:20)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:601)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120)
このハックは Lombok で使用され、注釈 @SneakyThrow を使用して、チェック済みの例外を宣言せずにスローできるようにします。
型消去と関係があることは知っていますが、ハックのすべての部分を理解しているかどうかはわかりません。
編集:
を挿入できることInteger
、List<String>
およびチェックされた/チェックされていない例外の区別がコンパイル時の機能であることを知っています。
コンパイラのList
ような非ジェネリック型からジェネリック型にキャストすると、警告が生成されます。ただし、上記のコードのList<XXX>
ようにジェネリック型に直接キャストすることはあまり一般的ではありません。(T) ex
必要に応じて、私にとって奇妙に思える部分は、JVM 内で aList<Dog>
とList<Cat>
同じように見えることを理解していることですが、上記のコードは、最終的に型 Cat の値を型 Dog または型の変数に割り当てることもできることを意味しているようですそんな感じ。