Java中反射机制和【Class.forName、实例对象.class(属性)、实例对象getClass()的区别】【符号引用及实例引用】

一、Java的反射机制
  每个Java程序执行前都必须经过编译、加载、连接、和初始化这几个阶段,后三个阶段如下图:
  其中
Java中反射机制和Class.forName、实例对象.class(属性)、实例对象getClass()的区别


  i、加载是指将编译后的java类文件(也就是.class文件)中的二进制数据读入内存,并将其放在运行时数据区的方法区内,然后再堆区创建一个Java.lang.Class对象,用来封装类在方法区的数据结构。即加载后最终得到的是Class对象,并且更加值得注意的是:该Java.lang.Class对象是单实例的,无论这个类创建了多少个对象,他的Class对象时唯一的!!!!。 而 加载并获取该Class对象可以通过三种途径:
     Class.forName(类的全路径)、实例对象.class(属性)、实例对象getClass()
。关于他们的区别将在下面讲到!!!
   ###另外 ,类加载时类中的静态代码块会得到执行(详见前一篇博客:Class.forName()加载JDBC驱动)

  ii、在 连接和初始化阶段,其实静态变量经过了两次赋值:第一次是静态变量类型的默认值;第二次是我们真正赋给静态变量的值。
  iii、Java对类的使用分为两种方式:主动使用和被动使用。其中主动使用如下图:
  Java中反射机制和Class.forName、实例对象.class(属性)、实例对象getClass()的区别而类的初始化时机正是java程序对类的首次主动使用,除了以上6中方式,其他对类的使用都是被动使用,都不会导致类的初始化。 并且应该注意以下几个方面:
 Java中反射机制和Class.forName、实例对象.class(属性)、实例对象getClass()的区别


Java中反射机制和Class.forName、实例对象.class(属性)、实例对象getClass()的区别

在这里可以看出;接口的两重性:可以把接口当做类(因为在接口中有静态变量时,他可以被初始化);接口就是接口,和类无关(接口中 没有构造方法,所以不能被初始化)

 二、Class.forName、实例对象.class(属性)、实例对象getClass()的区别

  1、相同点:
   通过这几种方式,得到的都是Java.lang.Class对象(这个是上面讲到的 类在加载时获得的最终产物)
  例如:
  

package demo;
public class A
{
    public static void main(String[] args) throws Exception
    {
      System.out.println(A.class);//通过类.class获得Class对象

       A a = new A();
      System.out.println(a.getClass());//通过 实例名.getClass()获得Class对象
      System.out.println(Class.forName("demo.A"));//通过Class.forName(全路径)获得Class对象
          System.out.println(".................................");
      System.out.println(a);//使用不同的方式创建对象
      System.out.println(A.class.newInstance());
      System.out.println(a.getClass().newInstance());
      System.out.println(Class.forName("demo.A").newInstance());  
    }
}


结果:class demo.A
     class demo.A
     class demo.A   (这里也可以得到一个类的Class对象是唯一的)
.......................................
     demo.A@de6ced
     demo.A@c17164
     demo.A@1fb8ee3
     demo.A@61de33

 2、区别:
1)Class cl=A.class; JVM将使用类A的类装载器,将类A装入内存(前提是:类A还没有装入内存),不对类A做类的初始化工作.返回类A的Class的对象

2)Class cl=对象引用o.getClass();返回引用o运行时真正所指的对象(因为:儿子对象的引用可能会赋给父对象的引用变量中)所属的类的Class的对象 

3)Class.forName("类名"); JAVA人都知道.装入类A,并做类的初始化

 附:

从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用Class对象的newInstance()方法的时候,就必须保证:1、这个 类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。 

现在可以看出,Class对象的newInstance()(这种用法和Java中的工厂模式有着异曲同工之妙)实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。 

最后用最简单的描述来区分new关键字和newInstance()方法的区别: 
newInstance: 弱类型。低效率。只能调用无参构造。 
new: 强类型。相对高效。能调用任何public构造。

三、JVM的符号引用和实例引用

在JVM中类加载过程中,在解析阶段,Java虚拟机会把类的二级制数据中的符号引用替换为直接引用。

1.符号引用(Symbolic References):

  符号引用以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能够无歧义的定位到目标即可。例如,在Class文件中它以CONSTANT_Class_info、CONSTANT_Fieldref_info、CONSTANT_Methodref_info等类型的常量出现。符号引用与虚拟机的内存布局无关,引用的目标并不一定加载到内存中。在Java中,一个java类将会编译成一个class文件。在编译时,java类并不知道所引用的类的实际地址,因此只能使用符号引用来代替。比如org.simple.People类引用了org.simple.Language类,在编译时People类并不知道Language类的实际内存地址,因此只能使用符号org.simple.Language(假设是这个,当然实际中是由类似于CONSTANT_Class_info的常量来表示的)来表示Language类的地址。各种虚拟机实现的内存布局可能有所不同,但是它们能接受的符号引用都是一致的,因为符号引用的字面量形式明确定义在Java虚拟机规范的Class文件格式中。

2.直接引用:

 直接引用可以是

(1)直接指向目标的指针(比如,指向“类型”【Class对象】、类变量、类方法的直接引用可能是指向方法区的指针)

(2)相对偏移量(比如,指向实例变量、实例方法的直接引用都是偏移量)

(3)一个能间接定位到目标的句柄

直接引用是和虚拟机的布局相关的,同一个符号引用在不同的虚拟机实例上翻译出来的直接引用一般不会相同。如果有了直接引用,那引用的目标必定已经被加载入内存中了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值