Uploaded image for project: 'JDK'
  1. JDK
  2. JDK-8024775

ClassLoader.loadClass throws NoClassDefFoundError with similar named class

XMLWordPrintable

    • Icon: Bug Bug
    • Resolution: Not an Issue
    • Icon: P4 P4
    • tbd
    • 6u37
    • core-libs

      FULL PRODUCT VERSION :
      >java -version
      java version " 1.6.0_37 "
      Java(TM) SE Runtime Environment (build 1.6.0_37-b06)
      Java HotSpot(TM) 64-Bit Server VM (build 20.12-b01, mixed mode)


      ADDITIONAL OS VERSION INFORMATION :
      Windows Server 2008 R2 Standard

      A DESCRIPTION OF THE PROBLEM :
      Given an existing class named 'ClassToLoad', calling classloader.loadClass( " CLASSToLoad " ) throws a NoClassDefFoundError instead of a ClassNotFoundException. [Note:the incorrect name because of caps case letters].

       This may apply on O/S that have case insensitive file naming (so the file is found but fails linking - which is case sensitive).

       I really feel this is inconsistent and breaches the 'principle of least surprise'. If I try to load a class which doesn't exist I expect a ClassNotFoundException.



      STEPS TO FOLLOW TO REPRODUCE THE PROBLEM :
      Please see the source add later in this form

      EXPECTED VERSUS ACTUAL BEHAVIOR :
      EXPECTED -
      If I call classloader.loadClass with the name of a class that doesn't exist, I expect a ClassNotFoundException
      ACTUAL -
      If I call classloader.loadClass with the name of a class that doesn't exist, but the name is similar to one that does exist (only differeing in the case some or all of the of characters) I get a NoClassDefFoundError

      ERROR MESSAGES/STACK TRACES THAT OCCUR :
      //
      // Output from running the attached source code ...
      //

      Successfully loaded an existing class
      Failed to load non-existent class 1 (expected behaviour)
      Exception in thread " main " java.lang.NoClassDefFoundError: CLASSToLoad (wrong name: ClassToLoad)
      at java.lang.ClassLoader.defineClass1(Native Method)
      at java.lang.ClassLoader.defineClassCond(Unknown Source)
      at java.lang.ClassLoader.defineClass(Unknown Source)
      at java.security.SecureClassLoader.defineClass(Unknown Source)
      at java.net.URLClassLoader.defineClass(Unknown Source)
      at java.net.URLClassLoader.access$000(Unknown Source)
      at java.net.URLClassLoader$1.run(Unknown Source)
      at java.security.AccessController.doPrivileged(Native Method)
      at java.net.URLClassLoader.findClass(Unknown Source)
      at java.lang.ClassLoader.loadClass(Unknown Source)
      at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
      at java.lang.ClassLoader.loadClass(Unknown Source)
      at Test.main(Test.java:36)

      REPRODUCIBILITY :
      This bug can be reproduced always.

      ---------- BEGIN SOURCE ----------
      public class Test {

          public static void main(String[] args) {
      // normal conditions
              try {
                  Test.class.getClassLoader().loadClass("ClassToLoad");
                  System.out.println(" Successfully loaded an existing class ");
              } catch (ClassNotFoundException notExpected) {
      // should NOT get here
                  System.out.println(" Failed to load an existing class ");
                  System.exit(-1);
              }


      // expected failure conditions
              try {
                  Test.class.getClassLoader().loadClass("NonExistingClass");
              } catch (ClassNotFoundException expected) {
                  System.out.println(" Failed to load non-existent class 1 (expected behaviour) ");
              }


      // inconsistent failure condition (a Bug in my opinion)
              try {
                  Test.class.getClassLoader().loadClass("CLASSToLoad");
              } catch (ClassNotFoundException expected) {
                  System.out.println(" Failed to load non-existent class 2 (desired behaviour) ");
              }
          }
      }

      //
      // sample class used for the above (requires separate file)
      //
      public class ClassToLoad
      {

      }

      ---------- END SOURCE ----------

      CUSTOMER SUBMITTED WORKAROUND :
      In a generic situtation, there is no work around. In the wild, an application that performs such loading is likely to have a unbounded set of possible inputs. It would experience NoClassDefFoundErrors where it would only expect ClassNotFoundExceptions (although one might argue that any code loading foreign classes this was should defend itself against linkage errors anyway).

       I am using this is a Java program that uses reflection to check API compiance against a language and platform neutral expression of an API and therefore I have bounded input (I can treat the naming of the problematic class as a special case)

            syu Seungrang Yu
            coffeys Sean Coffey
            Votes:
            1 Vote for this issue
            Watchers:
            3 Start watching this issue

              Created:
              Updated:
              Resolved: