Skip to content

Commit c9336d5

Browse files
committed
[J17] Add intermodule class loading example
Manual Loading: Few Modules (Separately - wtih deps).
1 parent 01a340a commit c9336d5

File tree

12 files changed

+221
-0
lines changed

12 files changed

+221
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
plugins {
2+
id 'java-library'
3+
}
4+
5+
group = 'com.habr.j17.manual.fewmodules.separately.nodeps.dependency'
6+
version = '1.0'
7+
8+
compileJava {
9+
java {
10+
toolchain {
11+
languageVersion = JavaLanguageVersion.of(17)
12+
}
13+
}
14+
}
15+
16+
sourceSets {
17+
main.java.srcDirs = ['src']
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
package com.habr.j17.manual.fewmodules.separately.withdeps.dependency;
2+
3+
public class Dog {
4+
public void talk() {
5+
System.out.println("Woof");
6+
}
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module manual.fewmodules.separately.withdeps.dependency {
2+
exports com.habr.j17.manual.fewmodules.separately.withdeps.dependency;
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
plugins {
2+
id 'java'
3+
id 'application'
4+
}
5+
6+
group = 'com.habr.j17.manual.fewmodules.separately.withdeps.loadable'
7+
version = '1.0'
8+
9+
compileJava {
10+
java {
11+
toolchain {
12+
languageVersion = JavaLanguageVersion.of(17)
13+
}
14+
}
15+
}
16+
17+
application {
18+
mainClassName = 'com.habr.j17.manual.fewmodules.separately.withdeps.loadable.Main'
19+
}
20+
21+
dependencies {
22+
implementation project(":java-17:manual:few-modules:separately:with-deps:dependency")
23+
}
24+
25+
sourceSets {
26+
main.java.srcDirs = ['src']
27+
}
28+
29+
jar {
30+
manifest {
31+
attributes(
32+
"Main-Class": application.mainClass
33+
)
34+
}
35+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
package com.habr.j17.manual.fewmodules.separately.withdeps.loadable;
2+
3+
public class Cat {
4+
public void talk() {
5+
System.out.println("Meow");
6+
}
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
package com.habr.j17.manual.fewmodules.separately.withdeps.loadable;
2+
3+
import com.habr.j17.manual.fewmodules.separately.withdeps.dependency.Dog;
4+
5+
public class Main {
6+
public static void main(String[] args) {
7+
Cat cat = new Cat();
8+
Dog dog = new Dog();
9+
10+
System.out.println("[loadable] Main Class ClassLoader is " + Main.class.getClassLoader());
11+
System.out.println("[loadable] Cat Class ClassLoader is " + Cat.class.getClassLoader());
12+
System.out.println("[dependency] Dog Class ClassLoader is " + Dog.class.getClassLoader());
13+
14+
System.out.println("[loadable] Main Class Module is " + Main.class.getModule());
15+
System.out.println("[loadable] Cat Class Module is " + Cat.class.getModule());
16+
System.out.println("[dependency] Dog Class Module is " + Dog.class.getModule());
17+
18+
cat.talk();
19+
dog.talk();
20+
}
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
module manual.fewmodules.separately.withdeps.loadable {
2+
requires manual.fewmodules.separately.withdeps.dependency;
3+
4+
exports com.habr.j17.manual.fewmodules.separately.withdeps.loadable;
5+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
plugins {
2+
id 'java'
3+
id 'application'
4+
}
5+
6+
group = 'com.habr.j17.manual.fewmodules.separately.withdeps.loading'
7+
version = '1.0'
8+
9+
compileJava {
10+
java {
11+
toolchain {
12+
languageVersion = JavaLanguageVersion.of(17)
13+
}
14+
}
15+
}
16+
17+
application {
18+
mainClassName = 'com.habr.j17.manual.fewmodules.separately.withdeps.loading.Main'
19+
}
20+
21+
sourceSets {
22+
main.java.srcDirs = ['src']
23+
}
24+
25+
jar {
26+
manifest {
27+
attributes(
28+
"Main-Class": application.mainClass
29+
)
30+
}
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
package com.habr.j17.manual.fewmodules.separately.withdeps.loading;
2+
3+
import java.io.File;
4+
import java.io.IOException;
5+
import java.io.InputStream;
6+
7+
public class CustomClassLoader extends ClassLoader {
8+
private final String PKG_PREFIX = "com.habr";
9+
10+
@Override
11+
public Class<?> loadClass(String name) throws ClassNotFoundException {
12+
Class<?> c = findLoadedClass(name);
13+
if (c != null)
14+
return c;
15+
16+
if (name.startsWith(PKG_PREFIX)) {
17+
c = findClass(name);
18+
if (c != null) {
19+
System.out.println("CCL: Loading " + name);
20+
return c;
21+
}
22+
}
23+
24+
System.out.println("CCL: Delegating " + name);
25+
return super.loadClass(name);
26+
}
27+
28+
@Override
29+
protected Class<?> findClass(String name) throws ClassNotFoundException {
30+
String classFile = name.replace('.', File.separatorChar) + ".class";
31+
System.out.println("CCL: Finding " + name);
32+
try (InputStream inputStream = getResourceAsStream(classFile)) {
33+
if (inputStream == null)
34+
throw new ClassNotFoundException();
35+
36+
byte[] bytecode = inputStream.readAllBytes();
37+
return defineClass(name, bytecode, 0, bytecode.length);
38+
} catch (IOException e) {
39+
throw new ClassNotFoundException();
40+
}
41+
}
42+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
package com.habr.j17.manual.fewmodules.separately.withdeps.loading;
2+
3+
import java.lang.module.Configuration;
4+
import java.lang.module.ModuleFinder;
5+
import java.lang.reflect.Method;
6+
import java.nio.file.Path;
7+
import java.nio.file.Paths;
8+
import java.util.Set;
9+
10+
public class Main {
11+
public static void main(String[] args) {
12+
try {
13+
Path loadableModulePath = Paths.get("../../../loadable/build/libs/loadable-1.0.jar");
14+
String loadableModuleName = "manual.fewmodules.separately.withdeps.loadable";
15+
Path dependencyModulePath = Paths.get("../../../dependency/build/libs/dependency-1.0.jar");
16+
String dependencyModuleName = "manual.fewmodules.separately.withdeps.dependency";
17+
18+
String mainClassName = "com.habr.j17.manual.fewmodules.separately.withdeps.loadable.Main";
19+
20+
ModuleLayer layer = createLayer(
21+
loadableModulePath, loadableModuleName,
22+
dependencyModulePath, dependencyModuleName
23+
);
24+
ClassLoader loadableCL = layer.findLoader(loadableModuleName);
25+
Class<?> mainClass = loadableCL.loadClass(mainClassName);
26+
27+
System.out.println("[loading] Main Class Module is " + Main.class.getModule());
28+
System.out.println("[loading] Main Class ClassLoader is " + Main.class.getClassLoader());
29+
System.out.println("ClassLoader for [loadable] module is " + loadableCL);
30+
31+
Method mainMethod = mainClass.getMethod("main", String[].class);
32+
mainMethod.invoke(null, (Object) args);
33+
} catch (Exception e) {
34+
throw new RuntimeException(e);
35+
}
36+
}
37+
38+
private static ModuleLayer createLayer(Path mp1, String mn1, Path mp2, String mn2) {
39+
ModuleFinder finder = ModuleFinder.of(mp1, mp2);
40+
41+
ModuleLayer bootLayer = ModuleLayer.boot();
42+
Configuration parentConfig = bootLayer.configuration();
43+
Configuration config = parentConfig.resolve(finder, ModuleFinder.of(), Set.of(mn1, mn2));
44+
return bootLayer.defineModulesWithOneLoader(config, new CustomClassLoader());
45+
}
46+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
module manual.fewmodules.separately.withdeps.loading {
2+
}

settings.gradle

+4
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,10 @@ include 'java-17:manual:few-modules:separately:no-deps:loadable'
1515
include 'java-17:manual:few-modules:separately:no-deps-hack:loading'
1616
include 'java-17:manual:few-modules:separately:no-deps-hack:loadable'
1717

18+
include 'java-17:manual:few-modules:separately:with-deps:loading'
19+
include 'java-17:manual:few-modules:separately:with-deps:loadable'
20+
include 'java-17:manual:few-modules:separately:with-deps:dependency'
21+
1822
include 'java-17:auto:one-module:java8style'
1923
include 'java-17:auto:one-module:java17style'
2024

0 commit comments

Comments
 (0)