Class loading

1.1. In Java code, type loading, linking, and initialization are all done during program execution.

Here is:



1.2. Type loading — What do we mean by type?

A: Type refers to our Java source code through a compiled class file.

1.3. What are the sources of types?

(1) Local disk

(2) Network download, class file

(3) Load the class file under war, jar

(4) Read from a dedicated database, class file (rare)

(5) Dynamically compile Java source files into class files

1) Dynamic proxies typically generate class files at runtime

2) Our JSP will be converted into servlets, while our Serlvet is a Java file that will be compiled into a class file

1.4. By what? (Class loader)



1.5. Classification of class loading and various loading responsibilities and hierarchy

(1) System level

1) Start the class loader

2) Extend the class loader

3) System class loader (App class loader)

(2) User level

Custom ClassLoader (inherits our ClassLoader)

(3) Hierarchy



Class loaders follow our parent delegate model when loading our classes

In the parent delegation mechanism, each loader forms a tree structure according to the parent-child relationship. In addition to the root loader, each loader has only one parent loader



1, source code analysis:

1 protectedClass<? > loadClass(String name,boolean resolve)
2  throws ClassNotFoundException
3 {
4  synchronized (getClassLoadingLock(name)) {
5  // Check if the current class object is loaded and return if it is loaded
6Class<? > c = findLoadedClass(name);7  if (c == null) {
8  long t0 = System.nanoTime();
9  try {
10  // Check whether the current classLoader has a parent class
11  // If there is a parent class
12  if(parent ! =null) {
13  // Call the parent loadClass
14  c = parent.loadClass(name, false);
15  } else {// If there is no parent class, the current classLoader is extClassLoader
16  // The startup class is called to determine if it has been loaded
17  c = findBootstrapClassOrNull(name);
18  }
19  } catch (ClassNotFoundException e) {
20  // ClassNotFoundException thrown if class not found
21  // from the non‐null parent class loader
22  }
23  // The app Ext Boot has not been loaded to the target location
24  if (c == null) {
25  // If still not found, then invoke findClass in order
26  // to find the class.
27  long t1 = System.nanoTime();
28  // Delegate our subclass's classLoader to find it
29  c = findClass(name);
30
31  // this is the defining class loader; record the stats
32Sun. Misc. PerfCounter. GetParentDelegationTime (). The addTime (t1 ‐ t0);33  sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
34  sun.misc.PerfCounter.getFindClasses().increment();
35  }
36  }
37  if (resolve) {
38  resolveClass(c);
39  }
40  return c;
41  }
42 }Copy the code

2. Flowchart of parental delegation model loading

3. Benefits of the parent delegate model for classloaders:

The java.lang.object class is the parent of all classes, so our program will load the java.lang.object class into memory during runtime, if the java.lang.object class can be loaded by our custom class loader. There will be multiple classes of Object in the JVM, and these classes are incompatible.

So the parental delegation model ensures the safety of types in the Java core class library.

With the parent delegate model, our Java core class library classes must be loaded by our startup class loader, which ensures that only one copy of our core class library exists in the JVM and does not give custom class loaders to load our core class library classes.

According to our example, a class can be loaded by multiple class loaders, and there can be multiple versions of class objects in JVM memory that are incompatible.

4, how to write a custom ClassLoader (according to the doc of ClassLoader)



(1) our custom ClassLoader must inherit the ClassLoader

(2) We must have the findClass(String name) method

1 /**
2  * Finds the class with the specified <a href="#name">binary name</a>.
3  * This method should be overridden by class loader implementations that
4  * follow the delegation model for loading classes, and will be invoked b
y
5  * the {@link #loadClass <tt>loadClass</tt>} method after checking the
6  * parent class loader for the requested class. The default implementatio
n
7  * throws a <tt>ClassNotFoundException</tt>.
8  *
9  * @param name
10  * The <a href="#name">binary name</a> of the class
11  *
12  * @return The resulting <tt>Class</tt> object
13  *
14  * @throws ClassNotFoundException
15  * If the class could not be found
16  *
17  * @since18 * / 1.2
19 protectedClass<? > findClass(String name)throws ClassNotFoundException
{
20  throw new ClassNotFoundException(name);
21 }Copy the code

LoadClassData (String name) loadClassData(String name)

1 /** 2 * Created by SMLZ on 2019/10/22.4 */
5 public class TulingClassLoader extends ClassLoader {
6
7  private final static String fileSuffixExt = ".class";
8
9  private String classLoaderName;
10
11  private String loadPath;
12
13  public void setLoadPath(String loadPath) {
14  this.loadPath = loadPath;
15  }
16
17  public TulingClassLoader(ClassLoader parent, String classLoaderName) {
18  /** 19 * Specifies the parent of the current class loader 20 */
21  super(parent);
22  this.classLoaderName = classLoaderName;
23  }
24
25  public TulingClassLoader(String classLoaderName) {
26  /** 27 * Use the appClassLoader as the loader for this class 28 */
29  super(a);30  this.classLoaderName = classLoaderName;
31  }
32
33  public TulingClassLoader(ClassLoader classLoader) {
34  super(classLoader);
35  }
36
37  Create a binary name for our class 39 *@author:smlz
40  * @paramName: binary name of the class 41 *@return42 * :@exception43 * :@date: 2019/10/22 14:42 44 * /
45  private byte[] loadClassData(String name) {
46  byte[] data = null;
47  ByteArrayOutputStream baos = null;
48  InputStream is = null;
49
50  try {
51  name = name.replace("."."\ \");
52  String fileName = loadPath+name+fileSuffixExt;
53  File file = new File(fileName);
54  is = new FileInputStream(file);
55
56  baos = new ByteArrayOutputStream();
57  int ch;
58  while (‐1! = (ch = is.read())){59  baos.write(ch);
60  }
61  data = baos.toByteArray();
62  }catch (Exception e) {
63  e.printStackTrace();
64  }finally {
65  try{
66  if(null! = baos) {67  baos.close();
68  }
69  if(null! =is) {70  is.close();
71  }
72  }catch (Exception e) {
73  e.printStackTrace();
74  }
75  }
76
77  return data;
78  }
79
80  protectedClass<? > findClass(String name)throws ClassNotFoundException
{
81  byte[] data = loadClassData(name);
82  System.out.println("TulingClassLoader loads our class :===>"+name);
83  return defineClass(name,data,0,data.length);
84  }
85 }Copy the code

(4) Special note: The default parent of our custom class loader is our system AppClassLoader

Code evidence:

1 public TulingClassLoader(String classLoaderName) {
2  /** 3 * Use the appClassLoader as the loader for this class
5  super(a);6  this.classLoaderName = classLoaderName;
7 }
8
9 // when super() is called
10 protected ClassLoader(a) {
11  // In this case, getSystemClassLoader() is our custom classloader
12  / / father
13  this(checkCreateClassLoader(), getSystemClassLoader());
14 }
15
16 private ClassLoader(Void unused, ClassLoader parent) {
17  this.parent = parent;
18  if (ParallelLoaders.isRegistered(this.getClass())) {
19  parallelLockMap = new ConcurrentHashMap<>();
20  package2certs = new ConcurrentHashMap<>();
21  domains =
22  Collections.synchronizedSet(new HashSet<ProtectionDomain>());
23  assertionLock = new Object();
24  } else {
25  / / no finer ‐ grained lock; lock on the classloader instance
26  parallelLockMap = null;
27  package2certs = new Hashtable<>();
28  domains = new HashSet<>();
29  assertionLock = this;
30  }
31 }Copy the code

5. How to prove that the parent of our custom class loader is the system class loader by experiment

(1) Copy our Person.class file to the specified disk directory. We also have our Person.class file in the classpath

3 * Created by SMLZ on 2019/11/12. 4 */ 5 Created by SMLZ on 2019/11/12 AppClassLoaderIsCustomerClassLoaderParent { 6 7 public static void main(String[] args) throws ClassNotFoundException { 8 Our system classloader loaded our Person 10, even though we loaded our Person.class 11 through the TulingClassLoader But since the parent delegate model will delegate our AppClassLoader to load the Person.class 12 and since our Classes directory has our Person.class 13 So our Person. The class is our AppClassLoader loaded. 14 15 16 17 = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = 18 If we delete the Person.class from the classpath, then our 19 TulingClassLoader tries to load our Person.class. But after we manually delete the Person.class from the classpath, our AppClassLo ader cannot load our Person.class. 22 **/ 23 test1(); 28 Public static void test1() throws ClassNotFoundException { 29 30 TulingClassLoader tulingClassLoader = new TulingClassLoader("tulingClas sLoader"); 31. / / set loading paths of 32 tulingClassLoader setLoadPath (" D: \ \ SMLZ \ \ "); 33 // Load our AppIsCustParentDemo 34 Class<? > targetClass = tulingClassLoader.loadClass("com.tuling.smlz.jv m.open.AppIsCustParentDemo"); 35 36 system.out. println("targetClass loaded by class loader... +targetClass.getCla ssLoader()); 37 38} 39}Copy the code

6. If the same Person. Class file is loaded by our different classloaders, our JVM memory will generate two corresponding Person class objects. It also makes sense that the two corresponding Class objects are invisible to each other (instance objects created by Class object reflection are incompatible with each other and cannot be transformed into each other)

1 public class Person {
2
3  private Person person;
4
5  public void setPerson(Object person) {
6  this.person = (Person) person;
7  }
8 }Copy the code

1 Public class Demo {2 // Remove the person. class from our ClassPath 3 public static void main(String[] args) throws ClassNotFoundException, Il legalAccessException, InstantiationException, NoSuchMethodException, Invoca tionTargetException { 4 5 TulingClassLoader classLoader1 = new TulingClassLoader("tulingClassLoade r1"); 6 classLoader1.setLoadPath("D:\\smlz\\"); 7 8 TulingClassLoader classLoader2 = new TulingClassLoader("tulingClassLoade r2"); 9 classLoader2.setLoadPath("D:\\smlz\\"); 10 11 // Load our Person 12 Class<? > class1 = classLoader1.loadClass 13 ("com.tuling.smlz.jvm.open.TheSameClassLoadedByDiffClassLoader.Person"); 14 system.out.println ("class loader for class1 :‐>"+ class.getClassloader ()); 15 16 Class<? > class2 = classLoader2.loadClass 17 ("com.tuling.smlz.jvm.open.TheSameClassLoadedByDiffClassLoader.Person"); 17 system.out.println ("class2 classloader: >"+ classloader ()); 19 20 System.out.println("class1==class2:"+(class1==class2)); Object person = class1. NewInstance (); 24 25 Object person2 = class2.newInstance(); 26 27 Method method = class1.getMethod("setPerson",Object.class); 29 method.invoke(person,person2); 30}} 31Copy the code

The overall delegation mechanism of the class loader and the namespace of the class loader

(1) The overall delegation mechanism of class loaders: For example, our Person class is loaded by our AClassLoader, then the Dog class referenced by our Person will be entrusted to our AClassLoader for loading

1 public class Person {2 3 public Person() {4 system.out.println ("Dog class is made by our class loader :‐ >"+ dog.class.getclassloader ()); 5 } 6 } 7 8 public class Dog { 9 } 10 11 public class MainTest { 12 13 public static void main(String[] args) { 14 15 Person person = new Person(); 16 system.out.println ("Person classLoader: >"+person.getClass().getClasslo ader()); 17 18} 19}Copy the code

(2) The namespace of the class loader

The class loader’s namespace consists of the class loader itself and the binary names loaded by all the parent loaders.

1) Two binary names are not allowed in the same namespace.

Two binary names can appear in different namespaces. The corresponding Class objects are not aware of each other, that is, the Class objects are of different types

The child loader can access the binary name in the parent loader’s namespace, and vice versa



Verify that classes loaded by the child loader can access classes loaded by the parent loader

Test environment: Our Person is loaded by our custom class loader, TulingClassLoader, which removes the Person.class from the classpath and copies it to the disk file. Dog is loaded by our AppClassLoader. We access Dog in Person.

1 public class Dog { 2 } 3 4 public class Person { 5 6 public Person() { 7 new Dog(); 8 System. Out. Println (" this Dog: ‐ ‐ > "+ t class. GetClassLoader ()); 9 } 10 11 } 12 13 public class TestDemo { 14 public static void main(String[] args) throws ClassNotFoundException, I llegalAccessException, InstantiationException { 15 TulingClassLoader classLoader = new TulingClassLoader("tulingClassLoade r"); 16 classLoader.setLoadPath("D:\\smlz\\"); 17 Class<? > clazz = classLoader.loadClass("com.tuling.smlz.jvm.open.classl oadernamespace.Person"); 18 clazz.newInstance(); Println ("Person classloader :"+clazz.getClassLoader()); 22 21}}Copy the code

9. How to prove that a class loaded by the parent cannot access a class loaded by the child

Test environment: put our person. class in C: ProgramFiles Java jdk1.8.0_131 jre classes Then our Person.class will be loaded by our launcher class loader, and our Dog class will be loaded by AppClassLoader, and our Person class that references our Dog class will throw an exception.

1 public class Dog {
2 }
3
4 public class Person {
5
6  public Person(a) {
7  new Dog();
8  System.out.println("This Dog: ‐ ‐ >"+ Dog.class.getClassLoader());
9  }
10 }
11
12
13 public class TestDemo {
14
15  public static void main(String[] args) throws IllegalAccessException, I
nstantiationException {
16
17
18  System.out.println("Person's classloader :"+Person.class.getClassLoader());
19
20  System.out.println("Dog's classloader :"+Dog.class.getClassLoader());
21
22Class<? > clazz = Person.class;23  clazz.newInstance();
24
25  }
26 }
27
28Running results:29Class loader for Person:null
30The class loader for Dog :sun.misc.Launcher$AppClassLoader@18b4aac2
31 Exception in thread "main" java.lang.NoClassDefFoundError: com/tuling/sm
lz/jvm/open/ParentClassLoaderNotAccessSonClassLoader/Dog
32  at com.tuling.smlz.jvm.open.ParentClassLoaderNotAccessSonClassLoader.Pe
rson.<init>(Person.java:11)
33  at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native
Method)
34  at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstruc
torAccessorImpl.java:62)
35  at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Delegating
ConstructorAccessorImpl.java:45)
36  at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
37  at java.lang.Class.newInstance(Class.java:442)
38  at com.tuling.smlz.jvm.open.ParentClassLoaderNotAccessSonClassLoader.Te
stDemo.main(TestDemo.java:16)Copy the code

A thread-context classloader that breaks the parent delegate model

Scenario: SPI application of JDBC interface technology.



The first active use of a class triggers initialization of the class.

1) Call static methods

2) Assign a static variable to obtain a static variable

3) reflection Class. Class.forname

4) New an object

5) Execute the main method

Initializing a child class initializes its parent class

The last

The latest Java interview questions manual for 2020 (200 + pages PDF document).