2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

NoClassDefFoundErrorの発生タイミングについて

Posted at

はじめに

世の中にはなぜか、コンパイル時には存在しているのに実行時には存在するかわからないようなクラスを取り扱わなければならないケースがあります。
存在していなければ正常にスキップし、NoClassDefFoundErrorClassNotFoundExceptionを発生させないようなコードとしなければなりません。
そのためにはどのタイミングでこれらの例外が発生するか知っておく必要があります。

前提条件

  • 対象のクラスやインターフェイスの存在は例外を発生させずクラス読み込みもしない方法でわかるものとします。
  • 今回は主にインターフェイスを扱います。
    • クラスでもほぼ一緒の結果になると思います。
  • この記事ではorg.apache.commons:commons-math4-core:4.0-beta1をコンパイル時にのみ読み込ませ、実行時には使用しないようにします。
    • このライブラリのorg.apache.commons.numbers.core.Additionを実装したりいろいろやります。
  • 例に出しているプログラムはパッケージ文など省略しているのでスタックトレースの行数とソースのコードは一致しません。

ケース(TL;DR)

番号 ケース エラーなし
1 何もせずにインターフェイスを実装したクラスのメソッドを呼ぶ
2 対象のインターフェイス(null)をフィールドに持つクラスのメソッドを呼ぶ
3 対象のインターフェイス(null)のフィールドにアクセスする
4 対象のインターフェイス(値あり)のフィールドにアクセスする
5 対象のインターフェイス(値あり)のフィールドを持つクラスにアクセスする ❌(条件あり)
6 対象のインターフェイスをメソッドの返り値(null)に持つクラスを初期化する
7 対象のインターフェイスをメソッドの返り値(コンストラクタの呼び出し)に持つクラスを初期化する
8 対象のインターフェイスをメソッドの返り値(staticでの呼び出し)に持つクラスを初期化する ✅(条件あり)
9 メソッド内のローカル変数で対象のインターフェイスを持っている場合
10 Ifのブロック内でローカル変数を持つ

ソースと結果

共通のコード

java ImplementNotExistInterface.java
import org.apache.commons.numbers.core.Addition;

class ImplementNotExistInterface implements Addition<Integer> {
    static void staticMethod() {
    }

    static Addition<Integer> returnAdditionInstance() {
        return new ImplementNotExistInterface();
    }

    static ImplementNotExistInterface returnImplInstance() {
        return new ImplementNotExistInterface();
    }

    static Object returnAsObject() {
        return new ImplementNotExistInterface();
    }

    @Override
    public Integer add(Integer a) {
        return null;
    }

    @Override
    public Integer zero() {
        return null;
    }

    @Override
    public Integer negate() {
        return null;
    }
}

1. 何もせずにインターフェイスを実装したクラスのメソッドを呼ぶ

ソース

java Test1.java

/**
 * Call method in a class implementing no-existing interface.
 */
public class Test1 {
    public static void main(String[] args) {
        System.out.println("Start Test1");
        // Here!
        ImplementNotExistInterface.staticMethod();
        System.out.println("End Test1");
    }
}

結果

Start Test1
Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.ClassLoader.defineClass1(Native Method)
	at java.base/java.lang.ClassLoader.defineClass(ClassLoader.java:1012)
	at java.base/java.security.SecureClassLoader.defineClass(SecureClassLoader.java:150)
	at java.base/jdk.internal.loader.BuiltinClassLoader.defineClass(BuiltinClassLoader.java:862)
	at java.base/jdk.internal.loader.BuiltinClassLoader.findClassOnClassPathOrNull(BuiltinClassLoader.java:760)
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClassOrNull(BuiltinClassLoader.java:681)
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:639)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	at jp.co.yumemi.koma.Test1.main(Test1.java:10)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 10 more

Process finished with exit code 1

2. 対象のインターフェイス(null)をフィールドに持つクラスのメソッドを呼ぶ

ソース

java Test2.java
import org.apache.commons.numbers.core.Addition;

public class Test2 {
    public static void main(String[] args) {
        System.out.println("Start Test2 instance");
        Test2HolderInstance.doNothing();
        System.out.println("End Test2 instance");

        System.out.println("Start Test2 static");
        Test2HolderStatic.doNothing();
        System.out.println("End Test2 static");
    }
}

class Test2HolderInstance {
    private final Addition<Integer> integerAddition = null;

    static void doNothing() {
    }
}

class Test2HolderStatic {
    private static final Addition<Integer> integerAddition = null;

    static void doNothing() {
    }
}

結果

Start Test2 instance
End Test2 instance
Start Test2 static
End Test2 static

Process finished with exit code 0

この場合はエラーになりません。

3. 対象のインターフェイス(null)のフィールドにアクセスする

コード

java Test3.java
import org.apache.commons.numbers.core.Addition;

public class Test3 {
    public static void main(String[] args) {
        System.out.println("Test3 access static start");
        if (Test3HolderStatic.integerAddition == null)
            System.out.println("Accessed Test3HolderStatic.integerAddition");
        System.out.println("Test3 access static end");

        System.out.println("Test3 access instance start");
        var i = new Test3HolderInstance();
        if (i.integerAddition == null)
            System.out.println("Accessed Test3HolderInstance.integerAddition");
        System.out.println("Test3 access instance end");
    }
}

class Test3HolderInstance {
    final Addition<Integer> integerAddition = null;

    static void doNothing() {
    }
}

class Test3HolderStatic {
    static final Addition<Integer> integerAddition = null;

    static void doNothing() {
    }
}

結果

Test3 access static start
Accessed Test3HolderStatic.integerAddition
Test3 access static end
Test3 access instance start
Accessed Test3HolderInstance.integerAddition
Test3 access instance end

Process finished with exit code 0

この場合もエラーにはなりません。
nullで初期化している場合にはアクセスしてもエラーにはならないので値を仮に埋めている際には注意してください。

4. 対象のインターフェイス(値あり)のフィールドにアクセスする

ソース

import org.apache.commons.numbers.core.Addition;

public class Test4 {
    public static void main(String[] args) throws Exception {
        System.out.println("Test4 access static start");
        try {
            if (Test4HolderStatic.integerAddition != null)
                System.out.println("Accessed Test4HolderStatic.integerAddition");
            System.out.println("Test4 access static end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }

        System.out.println("Test4 call static method");
        try {
            Test4HolderInstance.doNothing();
            System.out.println("Test4 call static method end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }

        System.out.println("Test4 access instance start");
        try {
            var i = new Test4HolderInstance();
            if (i.integerAddition != null)
                System.out.println("Accessed Test4HolderInstance.integerAddition");
            System.out.println("Test4 access instance end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
    }
}

class Test4HolderInstance {
    final Addition<Integer> integerAddition = new ImplementNotExistInterface();

    static void doNothing() {
    }

    {
        System.out.println("Test4HolderInstance initialize");
    }
}

class Test4HolderStatic {
    static final Addition<Integer> integerAddition = new ImplementNotExistInterface();

    static void doNothing() {
    }

    static {
        System.out.println("Test4HolderStatic static initialize");
    }
}

結果

Test4 access static start
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at jp.co.yumemi.koma.Test4.main(Test4.java:9)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 1 more
Test4 call static method
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at jp.co.yumemi.koma.Test4.main(Test4.java:19)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 1 more
Test4 access instance start
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at jp.co.yumemi.koma.Test4.main(Test4.java:28)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 1 more

Mainのクラスの呼び出しの手前のprintは呼ばれていますが、インスタンス初期化子や静的初期化子の中のprint文は呼ばれていません。
フィールドの宣言と初期化が初期化子の前に来ているからです。
またクラスにアクセスしたからエラーが発生したとも考えられるので、Test5で検証します。

5. 対象のインターフェイス(値あり)のフィールドを持つクラスにアクセスする

ソース

java Test5.java
import org.apache.commons.numbers.core.Addition;

public class Test5 {
    public static void main(String[] args) throws ReflectiveOperationException, InterruptedException {
        System.out.println("Test5 access static class start");
        try {
            System.out.println("Just access with .class");
            var c1 = Test5HolderStatic.class;
            System.out.println("Access via Class.forName, initialize=false");
            var c11 = Class.forName("jp.co.yumemi.koma.Test5HolderStatic",
                false, ClassLoader.getSystemClassLoader());
            System.out.println("Access via Class.forName, initialize=true");
            var c12 = Class.forName("jp.co.yumemi.koma.Test5HolderStatic",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test5 access static class end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            // Wait for stack trace printed
            Thread.sleep(500);
        }

        System.out.println("Test5 access instance class start");
        System.out.println("Just access with .class");
        var c2 = Test5HolderInstance.class;
        System.out.println("Access via Class.forName, initialize=false");
        var c21 = Class.forName("jp.co.yumemi.koma.Test5HolderInstance",
            false, ClassLoader.getSystemClassLoader());
        System.out.println("Access via Class.forName, initialize=true");
        try {
            var c22 = Class.forName("jp.co.yumemi.koma.Test5HolderInstance",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test5 access instance class end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }

        System.out.println("Test5 access instance class2 start");
        try {
            var c22 = Class.forName("jp.co.yumemi.koma.Test5HolderInstance2",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test5 access instance class2 end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
    }
}

class Test5HolderInstance {
    final Addition<Integer> integerAddition = new ImplementNotExistInterface();

    static void doNothing() {
    }

    {
        System.out.println("Test5HolderInstance initialize");
    }
}

class Test5HolderInstance2 {
    final Addition<Integer> integerAddition = ImplementNotExistInterface.returnAdditionInstance();

    static void doNothing() {
    }

    static {
        System.out.println("Test5HolderInstance2 static initialize");
    }
}

class Test5HolderStatic {
    static final Addition<Integer> integerAddition = new ImplementNotExistInterface();

    static void doNothing() {
    }

    static {
        System.out.println("Test5HolderStatic static initialize");
    }
}

結果

Test5 access static class start
Just access with .class
Access via Class.forName, initialize=false
Access via Class.forName, initialize=true
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.Class.forName0(Native Method)
	at java.base/java.lang.Class.forName(Class.java:467)
	at jp.co.yumemi.koma.Test5.main(Test5.java:15)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 3 more
Test5 access instance class start
Just access with .class
Access via Class.forName, initialize=false
Access via Class.forName, initialize=true
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.Class.forName0(Native Method)
	at java.base/java.lang.Class.forName(Class.java:467)
	at jp.co.yumemi.koma.Test5.main(Test5.java:32)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 3 more
Test5 access instance class2 start
Test5HolderInstance2 static initialize
Test5 access instance class2 end

.classでアクセスしたり、初期化しないでアクセスする分にはエラーにはなりません。
しかし初期化ありでアクセスするとstatic、インスタンスどちらで値を保持していてもエラーになります。
インスタンスフィールドで、初期化処理が走っていないにもかかわらずエラーになる点には注意が必要です。
特にコンストラクターを呼び出している場合にエラーになります。
staticメソッド経由でインスタンスを取得している場合にはエラーになりません。
初期化ありでアクセスとなるのはClass#forName(String)の使用やstaticメソッドの呼び出しがあります。

6. 対象のインターフェイスをメソッドの返り値(null)に持つクラスを初期化する

ソース

import org.apache.commons.numbers.core.Addition;

public class Test6 {
    public static void main(String[] args) throws Exception {
        System.out.println("Test6 start");

        System.out.println("Test6StaticHolder init start");
        var c1 = Test6StaticHolder.class;
        var c11 = Class.forName(c1.getCanonicalName(), true, c1.getClassLoader());
        Test6StaticHolder.doNothing();
        System.out.println("Test6StaticHolder init end");
        Thread.sleep(500);

        System.out.println("Test6MethodHolder init start");
        var c2 = Test6MethodHolder.class;
        var c21 = Class.forName(c2.getCanonicalName(), true, c2.getClassLoader());
        Test6MethodHolder.doNothing();
        System.out.println("Test6MethodHolder init end");
        System.out.println("Test6 end");

        System.out.println("Test6StaticHolder get methods start");
        var c1Methods = c1.getMethods();
        System.out.println("Test6StaticHolder get methods end");
        System.out.println("Test6StaticHolder getDeclaredMethods start");
        // here
        var c1Methods2 = c1.getDeclaredMethods();
        System.out.println("Test6StaticHolder getDeclaredMethods end");
    }
}

class Test6StaticHolder {
    static Addition<?> getAddition() {
        return null;
    }

    public static void doNothing() {
    }
}

class Test6MethodHolder {
    Addition<?> getAddition() {
        return null;
    }

    public static void doNothing() {
    }
}

結果

Test6 start
Test6StaticHolder init start
Test6StaticHolder init end
Test6MethodHolder init start
Test6MethodHolder init end
Test6 end
Test6StaticHolder get methods start
Test6StaticHolder get methods end
Test6StaticHolder getDeclaredMethods start
Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.Class.getDeclaredMethods0(Native Method)
	at java.base/java.lang.Class.privateGetDeclaredMethods(Class.java:3402)
	at java.base/java.lang.Class.getDeclaredMethods(Class.java:2504)
	at jp.co.yumemi.koma.Test6.main(Test6.java:27)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 4 more

Process finished with exit code 1

対象のインターフェイスを返す(ただしnull)メソッドを持っていてもそのメソッドの詳細を取得しようとしないかぎりエラーにはならないようです。
publicでなければClass#getMethods()を呼び出してもエラーにはなりません。
Class#getDeclaredMethods()で初めてエラーになります。

ここではnullを返すようにしていますが、例外を投げるなど対象のインターフェイスに関連しないことをしていればエラーにはなりません。

7. 対象のインターフェイスをメソッドの返り値(コンストラクタの呼び出し)に持つクラスを初期化する

ソース

package jp.co.yumemi.koma;

import org.apache.commons.numbers.core.Addition;

public class Test7 {
    public static void main(String[] args) throws Exception {
        System.out.println("Test7 start");

        System.out.println("Test7StaticHolder init(initialize=false) start");
        var c11 = Class.forName("jp.co.yumemi.koma.Test7StaticHolder",
            false, ClassLoader.getSystemClassLoader());
        System.out.println("Test7StaticHolder init(initialize=false) end");

        System.out.println("Test7StaticHolder class call start");
        try {
            var c22 = jp.co.yumemi.koma.Test7StaticHolder.class;
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
        System.out.println("Test7StaticHolder class call end");

        System.out.println("Test7StaticHolder init(initialize=true) start");
        try {
            // here
            var c21 = Class.forName("jp.co.yumemi.koma.Test7StaticHolder",
                true, ClassLoader.getSystemClassLoader());
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
        System.out.println("Test7StaticHolder init(initialize=true) end");

    }
}

class Test7StaticHolder {
    static Addition<?> getAddition() {
        return new ImplementNotExistInterface();
    }
}

結果

Test7 start
Test7StaticHolder init(initialize=false) start
Test7StaticHolder init(initialize=false) end
Test7StaticHolder class call start
Test7StaticHolder class call end
Test7StaticHolder init(initialize=true) start
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.Class.forName0(Native Method)
	at java.base/java.lang.Class.forName(Class.java:467)
	at jp.co.yumemi.koma.Test7.main(Test7.java:25)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 3 more
Test7StaticHolder init(initialize=true) end

Process finished with exit code 0

5のフィールドの時と同様、クラスの初期化処理を走らせると実際のメソッドを呼んでいないにもかかわらずエラーとなります。
存在しないインターフェイスを実装したクラスのコンストラクターの呼び出しがあるだけでエラーになるということです。

8. 対象のインターフェイスをメソッドの返り値(staticでの呼び出し)に持つクラスを初期化する

ソース

package jp.co.yumemi.koma;

import org.apache.commons.numbers.core.Addition;

public class Test8 {
    public static void main(String[] args) throws Exception {
        System.out.println("Test8 start");

        System.out.println("Test8StaticHolder1 init(initialize=true) start");
        try {
            var c21 = Class.forName("jp.co.yumemi.koma.Test8StaticHolder1",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test8StaticHolder1 init(initialize=true) end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }

        System.out.println("Test8StaticHolder2 init(initialize=true) start");
        try {
            // here
            var c21 = Class.forName("jp.co.yumemi.koma.Test8StaticHolder2",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test8StaticHolder2 init(initialize=true) end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }

        System.out.println("Test8StaticHolder3 init(initialize=true) start");
        try {
            var c21 = Class.forName("jp.co.yumemi.koma.Test8StaticHolder3",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test8StaticHolder3 init(initialize=true) end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
    }
}

class Test8StaticHolder1 {
    static Addition<?> getAddition() {
        return (Addition<?>) ImplementNotExistInterface.returnAsObject();
    }

    static Addition<?> getAddition2() {
        return ImplementNotExistInterface.returnAdditionInstance();
    }
}

class Test8StaticHolder2 {
    static Addition<?> getAddition() {
        return ImplementNotExistInterface.returnImplInstance();
    }
}

class Test8StaticHolder3 {
    static Object getAddition() {
        return ImplementNotExistInterface.returnImplInstance();
    }
}

結果

Test8 start
Test8StaticHolder1 init(initialize=true) start
Test8StaticHolder1 init(initialize=true) end
Test8StaticHolder2 init(initialize=true) start
java.lang.NoClassDefFoundError: org/apache/commons/numbers/core/Addition
	at java.base/java.lang.Class.forName0(Native Method)
	at java.base/java.lang.Class.forName(Class.java:467)
	at jp.co.yumemi.koma.Test8.main(Test8.java:22)
Caused by: java.lang.ClassNotFoundException: org.apache.commons.numbers.core.Addition
	at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:641)
	at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:188)
	at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:520)
	... 3 more
Test8StaticHolder3 init(initialize=true) start
Test8StaticHolder3 init(initialize=true) end

Process finished with exit code 0

このケースではTest8StaticHolder2の、返り値がAddition<?>で実際の値がImplementNotExistInterfaceのケースのみでエラーになりました。
返り値をObjectにしたり、static側をAddition<?>にしておけばエラーにはなりません。

9. メソッド内のローカル変数で対象のインターフェイスを持っている場合

ソース

import org.apache.commons.numbers.core.Addition;

public class Test9 {
    public static void main(String[] args) throws Exception {
        System.out.println("Start Test9");
        System.out.println("Test9Holder init(initialize=true) start");
        try {
            var c21 = Class.forName("jp.co.yumemi.koma.Test9Holder",
                true, ClassLoader.getSystemClassLoader());
            System.out.println("Test9Holder init(initialize=true) end");
        } catch (NoClassDefFoundError error) {
            error.printStackTrace(System.err);
            Thread.sleep(500);
        }
        System.out.println("End Test9");
    }

}

class Test9Holder {
    private static void hasAddition() {
        ImplementNotExistInterface anInterface = new ImplementNotExistInterface();
    }

    public static void use(Addition<?> addition) {
        System.out.println(addition.zero());
    }

    private static void h2() {
        Addition<Void> voidAddition = new Addition<>() {
            @Override
            public Void add(Void a) {
                return null;
            }

            @Override
            public Void zero() {
                return null;
            }

            @Override
            public Void negate() {
                return null;
            }
        };
    }
}

結果

Start Test9
Test9Holder init(initialize=true) start
Test9Holder init(initialize=true) end
End Test9

この場合はコンストラクターで呼び出していてもAdditionのインスタンスを直接作成しようとしてもエラーにはなりません。
もちろんstaticメソッド経由で値を返しても同様です。
引数にAdditionを持つメソッドがあってもいいようです。

10. Ifのブロック内でローカル変数を持つ

ソース

public class Test10 {
    public static void main(String[] args) {
        System.out.println("Test10 start");
        if (true) {
            System.out.println("In if block");
        } else {
            ImplementNotExistInterface a = new ImplementNotExistInterface();
        }
        System.out.println("Test10 end");
    }
}

結果

Test10 start
In if block
Test10 end

この場合もエラーになりませんでした。

まとめ

  • 基本的にクラスを初期化しないクラスへのアクセス方法であればエラーにはなりません
    • staticメソッドの呼び出しはクラスを初期化します
  • 存在しないインターフェイスを実装したクラスのコンストラクターの呼び出しがあるとエラーになることが多いです。
    • 特にフィールドで使用している際には実際に呼び出されなくてもクラスを初期化した段階でエラーになります。
    • 返り値としてnewして返すメソッドがあってもエラーになります。
    • 対象のクラスからstaticメソッド経由でインスタンスを取得する場合にはエラーになりません。
  • Ifなどのブロック内でコンストラクターを呼び出したりローカル変数を定義してもエラーになりません。

使用したソース
https://github.com/k-kojima-yumemi/symmetrical-sniffle

2
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?