科技行者

行者学院 转型私董会 科技行者专题报道 网红大战科技行者

知识库

知识库 安全导航

至顶网软件频道 Java对象创建过程

Java对象创建过程

  • 扫一扫
    分享文章到微信

  • 扫一扫
    关注官方公众号
    至顶头条

Java对象创建过程

作者:csdn 来源:csdn 2009年12月17日

关键字: JavaSE 问答 java

  • 评论
  • 分享微博
  • 分享邮件

Java对象创建过程

Java对象被创建有两种方式:第一种是通过new XX();第二种是Class.forName("XX").newInstance().我现在对这个流程的认识有些混乱.
    要生成一个类的对象,必须有两样东西被加载到内存,一种是Class对象;另一种是类,然后类对象才会被创建.由于Java是动态加载(类的静态成员被引用,类才会别加载),所以类加载的时候相对于上面的两种方式分别是在:第一种:new XX(),因为构造器方法也是静态方法,这种方式会调用构造器方法,所以此时类会被加载;第二种:Class.forName()首先会检查类有无被加载,没有则加载类.至此,两种方式的类都被加载.
    我的疑问是Class对象是何时被加载的,是和类一样吗?(我现在知道的关于Class对象的知识就是它是在.java文件被创建,被编译后产生了和类名字一样的.class文件.)是在类被加载之前还是之后,还有Class对象是怎么和类配合生成类对象的...

 

加载类就是加载Class对象。new XX();
Class.forName("XX").newInstance();
这两种效果一样,加载方式也一样,毫无区别,newInstance()实际上也是调用默认构造器方法的。
只是在用的场合不一样,前面一种用的很普遍。后面一种一般都是结合工厂模式应用的比较多。

 

这两种方式效果是一样的,但是稍微有以下两点小区别:
区别1:Class.forName的方式能实现动态地载和创建Class 对象,而NEW指定生成某个类的对象
例:String str = 你定义的字符串
    Class t = Class.forName(str);
    t.newInstance();
区别2:JAVA的工厂模式中常使用Class.forName方式创建对象
 
 
 
可能是我的表达能力不清楚.我想问的就是加载过程...到底一个对象是如何从无到有的.首先是编写一个后缀名为.java的类文件名叫Gun,经过编译后就生成了Gun.class文件.我的意思是类加载器(JVM的一部分)总得加载它们吧(它们代指Class对象和类Gun,每个类都对应一个Class对象).这个加载过程是怎么个样子的.一楼说的也只是说了表面.都是调用默认构造方法.中间这个调用过程还是要牵扯到加载Class对象,加载类,还有一些我还不明白的过程...我看过这么一句话:'当Class对象被载入内存,它就用来生成所有的对象'.就是说Gun.class被载入内存就是用来生成所有的Gun类的对象的...

 
 
关键是 static initialization
class A {
    //
    static {
        System.out.println("haha");
    }
}
这个东西类似常量的初始化。
第一种调用方法,VM 启动的时候,会去找所有被 import 了的类,如果找到了,这段代码就会被调用。也就是一启动就会调用。
第二种方法,如果你一直没有 import 这个类,vm 一直等到 Class.forName("XX").newInstance() 才会调。如果 static initialization 里面写了很多代码,又不想在不需要的时候调用它们,就只有第二种方法来。
 
 
说明一下加载的过程:
public class LoaderTest{
    public static void main(String args[])throws ClassNotFoundException{
    Class c;
  LoaderTest lt=new LoaderTest();
    c=Class.forName("LoaderTest");
    System.out.println(c);
    System.out.println(c.getName());
    System.out.println(c.getClassLoader());
    System.out.println(c.getClassLoader().getParent());
    System.out.println(c.getClassLoader().getParent().getParent());
}
}
class LoaderTest
LoaderTest
sun.misc.Launcher$AppClassLoader@19821f
sun.misc.Launcher$ExtClassLoader@addbf1
null
首先对于编译后的LoaderTest.class,必须要知道哪个加载器来加载,(下面是一种委托模式,及子类加载器都先调用父类加载器加载)首先调用AppClassLoader来加载,但AppClassLoader会让他继承的父类加载器ExtClassLoader加载LoaderTest.class,ExtClassLoader没有父类加载器(返回值为null),将LoaderTest.class转交给虚拟内核中的Bootstrap,由于Bootstrap只能加载java核心包中的类,所以告诉ExtClassLoader自己不能加载,ExtClassLoader是用来加载%JAVA_HOME%/jre/lib/ext/中的jar包中类,自己也不能加载LoaderTest.class,于是就告诉AppClassLoader自己不能加载该LoaderTest.class。由于AppClassLoader可以加载应用程序的启动态,可以加载该LoaderTest.class,
然后就像上面说的那样,加载时生成了一个个Class类的实例对象,该Class对象封装了 LoaderTest.class的字节码数据,也就是生成存储在内存中的可执行代码。我们可以通过Class.forName("LoaderTest").getName()获得这个Class对象的名称,通过Class.forName("LoaderTest").newInstance()来获得 LoaderTest这个类的对象。
 
    • 评论
    • 分享微博
    • 分享邮件
    邮件订阅

    如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。

    重磅专题
    往期文章
    最新文章