`

JAVA类加载以及反射机制

阅读更多
     JAVA中类文件加载是动态的。JVM指令是被封装在了. class文件里面,而.class文件的加载过程是动态的,也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类。这里所说的用到包括两种方式,第一种就是new一个对象的时候(这个时候要特别注意,当设计到多态的时候,就会有一点点变化,这时候编译器会做一些优化,这样以来当加载的时候会提前加载设计到多态的类,关于这一点下面有个例子(example 1)来说明。另一种就是当一个类的静态代码被调用的时候。
     
//example 1
// Zoo.java

abstract class Animal {

	Animal(){
	
		 System.out.println("Animal constructor");
	}
}

class Tiger extends Animal {

	Tiger(){
    	
		System.out.println("Tig constructor ");
	}
		
}


class Dog extends Animal {

	Dog(){
	
		System.out.println("Dog Constructor ");
	}
}



public class Zoo {

	  private Animal am; //Example 1.1
	  //private Dog am;     Example 1.2
		private Tiger tiger;
		
		Zoo(){
                          am = new Dog();
                          tiger = new Tiger();
		       			
		}
		
	
	 public static void main(String [] args){
	
	     System.out.println("new Zoo before");
              Zoo z = new Zoo();
	     System.out.println("new Zoo after ");
		
	 }
}

运行此测试程序结果如Example1.1图 :
Example 1.1



当我们注释掉Example.1.1行时,运行Example1.2行,结果如下:

Example 1.2





分析以上两图的运行结果我们可以看出:当我们将子类对象赋值给父类时,编译器会做一点优化,于是加载器在还没有new 子类对象的时候已经加载了父类以及子类(example1.1结果),当不存在多态的时候,我们可以看到是当要new Dog()的时候才会加载Dog以及父类。无论何种方式,在new之前,类确实已经加载到了内存中。
      

      JAVA为我们提供了两种动态机制。第一种是隐式机制。其实new一个对象和调用类的静态方法时,就是隐式机制在工作。第二种是显示机制。显示的机制又有两种策略(第一种是用java.lang.Class的forName(String str)方法,第二种是用java.lang.ClassLoader的loadClass())。
 
      第一种:利用forName方法
      当我们查API文档就会发现forName方法有两种形式。分别如下:
      public static Class<?> forName(String className)
                        throws ClassNotFoundException
    
     
      public static Class<?> forName(String name,
                               boolean initialize,
                               ClassLoader loader)
                        throws ClassNotFoundException

      先来说说第二种方法:第二个方法值得注意的就是第二个参数boolean initialize,如果我们把这个参数设置为false,那么当我们加载完类后就不会执行静态代码和静态的初始化动作。只有当我们new一个对象的时候才会初始化。而第三个参数是用来指明类的加载器的。
      如果查看java.lang.Class类的源代码,上述两种方法最终都会调用Class类中的私有的native方法forName0(),此方法的声明如下:
      private static native Class forName0(String name, boolean  init , ClassLoader loader)
                  throws ClassNotFoundException;

所以当我们调用Class.forName(name )时,其实是在方法内部调用了:
                  forName0(name, true, ClassLoader.getCallerClassLoader());
    当我们调用Class.forName(name,  initialize, loader )的时候,实际上此方法内部调用了:
                 forName0(name, initialize, loader);
下面看一个例子,如果方法中第二个参数为false的情况:
//example 2.1 
// Zoo.java

abstract class Animal {
  
  static {
  
  	System.out.println("Animal static code block ");
  }
	Animal(){
	
		 System.out.println("Animal constructor");
	}
}

class Tiger extends Animal {

	Tiger(){
    	
		System.out.println("Tig constructor ");
	}
		
}


class Dog extends Animal {

	Dog(){
	
		System.out.println("Dog Constructor ");
	}
}



public class Zoo {

	
	 public static void main(String [] args)throws Exception {
	
	     System.out.println("new Zoo before");
	     Zoo z = new Zoo();
	     Class c = Class.forName("Dog",false,z.getClass().getClassLoader());
	     System.out.println("initilize before ");
	     Animal dog = (Animal)c.newInstance();
            System.out.println("new Zoo after ");
		
	 }
}

Example 2.1
  


从上图可以看出来类加载完成后并没有立即执行静态初始化代码,而是到了实例化的时候才进行了静态初始化。有时候我们会说静态代码是在类第一次被加载时执行的,并且只执行一次。其实这是对与new一个对象,第一次访问类的静态代码以及第二个参数为true时而言的,对于动态的加载来说,如果forName方法的第二个参数设置为false,那么就是在实例化的时候才会执行静态初始化。当然默认情况下第二个参数是true.

第二种方法:利用Class对象获取的ClassLoader装载。

下面是一个简单的例子:

//Example 2.2
//Zoo.java
abstract class Animal {
  
  static {
  
  	System.out.println("Animal static code block ");
  }
	Animal(){
	
		 System.out.println("Animal constructor");
	}
}

class Tiger extends Animal {

	Tiger(){
    	
		System.out.println("Tig constructor ");
	}
		
}


class Dog extends Animal {

	Dog(){
	
		System.out.println("Dog Constructor ");
	}
}



public class Zoo {

	
	 public static void main(String [] args)throws Exception {
	
	     
	     Class c = Zoo.class;
	     ClassLoader loader = c.getClassLoader();
	     System.out.println("loader before");
	     Class dog = loader.loadClass("Dog");
	     System.out.println("instance before ");
	     Animal an = (Animal)dog.newInstance();
	     
	     
		
	 }
}


Example 2.2



从上图可以看出loader完成以后并没有立即进行静态代码的执行。只有当newInstance()的时候才执行静态初始化,这和把
public static Class forName(String name, boolean initialize,  ClassLoader loader)的第二个参数指定为false的情况完全一样。其实每当我们写完一个编译单元以后就会得到一个.calss文件,这个文件中就包含了该类的Class对象。JVM就是利用这个class对象来进行动态装载类的。

呵呵终于总结玩了。如果有什么地方理解有误的,请各位前辈和各位兄弟指点。谢谢。。
分享到:
评论

相关推荐

    Java类加载机制与反射-PPT

    Java的类加载机制:加载,连接,初始化。JAVA类加载器: Bootstrap ClassLoader : 根类加载器, Extension ClassLoader: 扩展类加载器, System ClassLoader : 系统类加载器, Java反射

    类加载机制及反射

    Java语言的类加载机制是其运行时环境的重要组成部分,它负责将类的字节码文件从磁盘、网络或其他数据源加载到JVM(Java虚拟机)中,并转化为可以执行的Java对象。类加载机制包括加载、验证、准备、解析和初始化五个...

    java面试题--反射机制

    动态加载类是Java反射机制的重要应用场景之一。通过`Class.forName()`方法或者`ClassLoader`的`loadClass()`方法,可以根据类名字符串在运行时加载类。 #### 七、操作成员 - **创建对象**:使用`Class`对象的`new...

    java类加载机制

    ### Java 类加载机制详解 Java 类加载机制是Java运行时环境的一个核心组成部分,它负责将编译后的Java字节码加载到JVM中,并确保Java应用程序能够正确地运行。类加载机制不仅涉及到类的加载、验证、准备、解析和...

    java注解_反射_字节码_类加载机制.zip

    通过阅读资料,你可以了解到注解的应用场景、反射的API用法、字节码的结构以及类加载机制的工作流程。而源代码分析将加深你对这些概念的实际运用。 总之,掌握Java注解、反射、字节码和类加载机制对于提升Java开发...

    反射机制和类加载机制学习总结

    反射机制和类加载机制是Java编程语言中的两个重要概念。反射提供了强大的运行时元数据操作能力,而类加载机制则确保了类的安全加载和动态管理。深入理解这两个机制对于Java开发者来说至关重要,尤其对于那些从事框架...

    Java反射动态加载实例类

    ### Java反射机制与动态加载实例类 在Java中,反射是一种强大的工具,允许程序在运行时检查和修改其结构和行为。通过反射,我们可以动态地加载类、创建对象、访问和修改字段、调用方法等。本文将深入探讨Java反射...

    Java反射机制 Java反射机制

    Java反射机制是Java语言的一个重要特性,它允许程序在运行时获取类的信息并操作对象。Java反射机制的主要作用包括:获取类的所有属性和方法、构造动态实例、调用类的方法等。通过反射,程序可以动态地创建对象和调用...

    java 类加载器 加密

    在“java 类加载器 加密”这个主题中,我们将探讨如何利用类加载器实现类的加密和解密,以及如何通过反射执行main方法。 首先,我们理解一下类加载器的工作原理。Java中的类加载器主要有三种:Bootstrap ...

    Java动态类加载机制研究及应用.pdf

    Java 反射机制允许 Java 程序在运行时动态地加载和调用类的方法。通过反射机制,可以实现动态地加载和执行 Java 类文件。 除了实现动态类加载机制外,还可以应用于网络教学环境中。通过开发基于 C/S 结构模式的...

    JAVA类加载机制与动态代理

    ### JAVA类加载机制与动态代理 #### 一、类加载机制 ##### 1.1 类加载的时机 类加载机制负责将描述类的数据从`.class`文件加载到内存,并进行必要的校验、转换解析和初始化,使之成为可以被Java虚拟机直接使用的...

    Java中类的反射机制

    在Java中,反射机制主要通过java.lang.Class类和相关的类库来实现,它使得我们能够在运行时动态地获取类的信息(如类名、属性、方法等)并能创建和调用对象。 1. **类的加载与Class对象** 当Java虚拟机(JVM)加载一...

    JAVA 类加载机制分析

    JAVA 类加载机制是Java平台核心特性之一,它关乎到程序的运行时环境和代码的动态加载。理解这一机制有助于开发者解决与对象创建、配置问题、应用程序发布等相关的问题。以下是关于JAVA 类加载机制的详细分析: 首先...

    Java中的反射机制

    Java中的反射机制为开发者提供了一种强大而灵活的方式来处理类信息。通过反射,可以实现诸如动态代理、框架开发等高级功能,同时也可以用于构建高度可扩展和可维护的软件系统。然而,需要注意的是,反射操作可能会...

    java加载类与反射机制

    Java 程序在运行期间可以动态加载、解析和使用一些在编译阶段并不确定的类型数据,这一机制被称为反射(Reflection)。反射库(reflection library)提供了一个非常丰富且精心设计的工具类,以便编写能够动态操纵Java...

    java中的类反射机制

    Java作为一种高级编程语言,支持反射机制,这使得开发者能够在运行时动态地访问和操作类、接口以及它们的成员。Java中的反射主要包括以下几个方面: 1. **Class对象**:每个加载进JVM的类都有对应的`Class`对象,该...

    JAVA反射机制的入门代码

    总的来说,"JAVA反射机制的入门代码"是初学者理解Java反射机制的好教材,通过它,你可以学习如何动态地操作Java类,提高代码的灵活性,并掌握处理`properties`文件的基础方法。在深入学习和实践中,你将进一步理解...

    JAVA反射机制与类的加载

    Java反射机制使得程序在运行时能够动态加载、解析和使用未知类型的类。这为开发者提供了强大的灵活性,比如在设计和运行时添加新类,或者动态查询和操作类的能力。通过反射,我们可以: 1. 加载运行时才能确定的...

Global site tag (gtag.js) - Google Analytics