mirror of
https://github.com/corda/corda.git
synced 2025-01-22 20:38:05 +00:00
453ceb42ab
Unlike the interpreter, the JIT compiler tries to resolve all the symbols referenced by a method when compiling that method. However, this can backfire if a symbol cannot be resolved: we end up throwing an e.g. NoClassDefFoundError for code which may never be executed. This is particularly troublesome for code which supports multiple APIs, choosing one at runtime. The solution is to defer to stub code for symbols which can't be resolved at JIT compile time. Such a stub will try again at runtime to resolve the needed symbol and throw an appropriate error if it still can't be found.
185 lines
4.4 KiB
Java
185 lines
4.4 KiB
Java
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.IOException;
|
|
|
|
public class LazyLoading {
|
|
private static boolean loadLazy;
|
|
|
|
private static void expect(boolean v) {
|
|
if (! v) throw new RuntimeException();
|
|
}
|
|
|
|
private static File findClass(String name, File directory) {
|
|
for (File file: directory.listFiles()) {
|
|
if (file.isFile()) {
|
|
if (file.getName().equals(name + ".class")) {
|
|
return file;
|
|
}
|
|
} else if (file.isDirectory()) {
|
|
File result = findClass(name, file);
|
|
if (result != null) {
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private static byte[] read(File file) throws IOException {
|
|
byte[] bytes = new byte[(int) file.length()];
|
|
FileInputStream in = new FileInputStream(file);
|
|
try {
|
|
if (in.read(bytes) != (int) file.length()) {
|
|
throw new RuntimeException();
|
|
}
|
|
return bytes;
|
|
} finally {
|
|
in.close();
|
|
}
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
Class c = new MyClassLoader(LazyLoading.class.getClassLoader()).loadClass
|
|
("LazyLoading$Test");
|
|
|
|
c.getMethod("test").invoke(null);
|
|
}
|
|
|
|
private static class MyClassLoader extends ClassLoader {
|
|
public MyClassLoader(ClassLoader parent) {
|
|
super(parent);
|
|
}
|
|
|
|
protected Class findClass(String name) throws ClassNotFoundException {
|
|
try {
|
|
return defineClass
|
|
(name, read
|
|
(LazyLoading.findClass
|
|
(name, new File(System.getProperty("user.dir")))));
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
public Class loadClass(String name) throws ClassNotFoundException {
|
|
if ("LazyLoading$Test".equals(name)) {
|
|
return findClass(name);
|
|
} else if ("LazyLoading$Lazy".equals(name)
|
|
|| "LazyLoading$Interface".equals(name))
|
|
{
|
|
if (loadLazy) {
|
|
return findClass(name);
|
|
} else {
|
|
throw new ClassNotFoundException();
|
|
}
|
|
} else {
|
|
return super.loadClass(name);
|
|
}
|
|
}
|
|
|
|
private Class defineClass(String name, byte[] bytes) {
|
|
return defineClass(name, bytes, 0, bytes.length);
|
|
}
|
|
}
|
|
|
|
private static class Test {
|
|
public static void test() {
|
|
doTest();
|
|
loadLazy = true;
|
|
doTest();
|
|
}
|
|
|
|
private static void doTest() {
|
|
if (loadLazy) {
|
|
// anewarray
|
|
Lazy[] array = new Lazy[1];
|
|
|
|
// new and invokespecial
|
|
Object lazy = new Lazy();
|
|
|
|
// checkcast
|
|
array[0] = (Lazy) lazy;
|
|
|
|
// instanceof
|
|
expect(lazy instanceof Lazy);
|
|
|
|
// invokeinterface
|
|
Interface i = array[0];
|
|
expect(i.interfaceMethod() == 42);
|
|
|
|
// invokestatic
|
|
expect(Lazy.staticMethod() == 43);
|
|
|
|
// invokevirtual
|
|
expect(array[0].virtualMethod() == 44);
|
|
|
|
// ldc
|
|
expect(Lazy.class == lazy.getClass());
|
|
|
|
// multianewarray
|
|
Lazy[][] multiarray = new Lazy[5][6];
|
|
multiarray[2][3] = array[0];
|
|
expect(multiarray[2][3] == array[0]);
|
|
|
|
// getfield
|
|
expect(array[0].intField == 45);
|
|
|
|
// getstatic
|
|
expect(Lazy.intStaticField == 46);
|
|
|
|
// putfield int
|
|
array[0].intField = 47;
|
|
expect(array[0].intField == 47);
|
|
|
|
// putfield long
|
|
array[0].longField = 48;
|
|
expect(array[0].longField == 48);
|
|
|
|
// putfield object
|
|
Object x = new Object();
|
|
array[0].objectField = x;
|
|
expect(array[0].objectField == x);
|
|
|
|
// putstatic int
|
|
array[0].intStaticField = 49;
|
|
expect(array[0].intStaticField == 49);
|
|
|
|
// putstatic long
|
|
array[0].longStaticField = 50;
|
|
expect(array[0].longStaticField == 50);
|
|
|
|
// putstatic object
|
|
Object y = new Object();
|
|
array[0].objectStaticField = y;
|
|
expect(array[0].objectStaticField == y);
|
|
}
|
|
}
|
|
}
|
|
|
|
private interface Interface {
|
|
public int interfaceMethod();
|
|
}
|
|
|
|
private static class Lazy implements Interface {
|
|
public static int intStaticField = 46;
|
|
public static long longStaticField;
|
|
public static Object objectStaticField;
|
|
|
|
public int intField = 45;
|
|
public long longField;
|
|
public Object objectField;
|
|
|
|
public int interfaceMethod() {
|
|
return 42;
|
|
}
|
|
|
|
public static int staticMethod() {
|
|
return 43;
|
|
}
|
|
|
|
public int virtualMethod() {
|
|
return 44;
|
|
}
|
|
}
|
|
}
|