Java中类加载过程全面解析

 更新时间:2017年11月24日 11:28:14   作者:moira33  
这篇文章主要介绍了Java中类加载过程全面解析,具有一定参考价值,需要的朋友可以了解下。

类文件加载的顺序

1、先加载执行父类的静态变量及静态初始化块(执行先后顺序按排列的先后顺序)

2、再加载执行本类的静态变量及静态初始化块

只要类没有被销毁,静态变量及静态初始化块只会执行1次,后续再对该类进行其他操作也不会再执行这两个步骤。

类实例创建过程

只有在调用new方法时才会创建类的实例

1、按照上面类文件加载的顺序(类已被加载则跳过此步)

2、父类的非静态变量及非静态初始化块

3、父类的构造方法

4、本类的非静态变量及非静态初始化块

5、本类的构造方法

4、类实例销毁时候,首先销毁子类部分,再销毁父类部分

静态方法和非静态方法都是被动调用

即系统不会自动调用执行。所以用户没有调用时都不执行,主要区别在于静态方法可以直接用类名直接调用(实例化对象也可以),而非静态方法只能先实例化对象后才能调用。

相关概念

static关键字:

是一个修饰符,用于修饰成员(成员变量和成员函数)

被修饰后的成员具备以下特点:

随着类的加载而加载(类一加载,静态数据就立即在内存中加载空间)

随着类的消失而消失,说明它的生命周期最长

优先于对象存在(对象消失了,static还在)

静态先存在,对象后存在

被所有对象所共享

节约内存空间

当成员被静态修饰后,除了可以被对象调用外,还可以直接被类名调用

写法:类名.静态成员

使用注意

静态方法只能访问静态成员(方法和变量)

非静态方法既可以访问静态也可以访问非静态

静态方法中不可以写this,super关键字

因为静态优先于对象存在,所以静态方法中不可以出现this

主函数是静态的

publicstaticvoidmain(String[]args){}

何时使用静态?

要从两方面下手:因为静态修饰的内容有成员变量和函数。

何时定义静态变量(类变量)

当对象中出现共享数据时,该数据被静态所修饰。对象中的特有数据要定义成非静态存在于堆内存中。

何时定义静态函数

当功能内部没有访问到非静态数据(对象特有数据),该功能可以定义成静态。

静态利弊

利:

1、对对象的共享数据单独空间的存储,节省空间。没有必要每个对象都存储一份。

2、可以直接被类名调用

弊:

1、生命周期过长

2、访问出现局限性(只能访问静态)

内存结构

Java程序在运行时,需要在内存中的分配空间。为了提高运算效率,有对空间进行了不同区域的划分,因为每一片区域都有特定的处理数据方式和内存管理方式。

栈内存

用于存储局部变量当数据使用完,所占空间会自动释放

堆内存

数组和对象(实体),通过new建立的实例都存放在堆内存中(成员变量随着对象的建立而建立,存在于对象所在的堆内存中)每一个实体都有内存地址值(变量通过地址引用)实体中的变量都有默认初始化值实体不再被使用,会在不确定的时间内被垃圾回收器回收(垃圾回收机制)

方法区,本地方法区,寄存器

验证

加载顺序 父类静态变量=1 父类非静态变量=1 子类静态变量=1 子类非静态变量=1
【父类调用父类静态方法】 Parent.pStaticMethod();      
父类静态初始化块一 2      
父类静态初始化块二 3      
父类静态方法 4      
【子类调用子类静态方法】 Child.cStaticMethod();      
子类静态初始化块一 5   2  
子类静态初始化块二 6   3  
子类静态方法 7   4  
【子类实例化】 Child c=new Child();      
父类非静态初始化块一 8 2    
父类非静态初始化块二 9 3    
父类构造方法 10 4    
子类非静态初始化块一 11 5 5 2
子类非静态初始化块二 12 6 6 3
子类构造方法 13 7 7 4
【父类实例化子类对象】 Parent p=new Child();      
父类非静态初始化块一 14 2    
父类非静态初始化块二 15 3    
父类构造方法 16 4    
子类非静态初始化块一 17 5 8 2
子类非静态初始化块二 18 6 9 3
子类构造方法 19 7 10 4
加载顺序 父类静态变量=1 父类非静态变量=1 子类静态变量=1 子类非静态变量=1
【子类实例化】 Child c=new Child();      
父类静态初始化块一 2      
父类静态初始化块二 3      
子类静态初始化块一 4   2  
子类静态初始化块二 5   3  
父类非静态初始化块一 6 2    
父类非静态初始化块二 7 3    
父类构造方法 8 4    
子类非静态初始化块一 9 5 4 2
子类非静态初始化块二 10 6 5 3
子类构造方法 11 7 6 4
【父类实例化子类对象】 Parent p=new Child();      
父类非静态初始化块一 12 2    
父类非静态初始化块二 13 3    
父类构造方法 14 4    
子类非静态初始化块一 15 5 7 2
子类非静态初始化块二 16 6 8 3
子类构造方法 17 7 9 4
【父类调用父类静态方法】 Parent.pStaticMethod();      
父类静态方法 18      
【子类调用子类静态方法】 Child.cStaticMethod();      
子类静态方法 19   10  

public class ClassTest {  
  public static void main (String args[]) {
    System.out.println("【子类实例化】|Child c=new Child();");   
    Child c=new Child();
    System.out.println("【父类实例化子类对象】|Parent p=new Child();");    
    Parent p=new Child();
    System.out.println("【父类调用父类静态方法】|Parent.pStaticMethod();");
    Parent.pStaticMethod();
    System.out.println("【子类调用子类静态方法】|Child.cStaticMethod();");
    Child.cStaticMethod();
  }
}
public class ClassTest2 {  
  public static void main (String args[]) {
    System.out.println("【父类调用父类静态方法】|Parent.pStaticMethod();");
    Parent.pStaticMethod();
    System.out.println("【子类调用子类静态方法】|Child.cStaticMethod();");
    Child.cStaticMethod();
    System.out.println("【子类实例化】|Child c=new Child();");   
    Child c=new Child();
    System.out.println("【父类实例化子类对象】|Parent p=new Child();");    
    Parent p=new Child();
  }
}
public class Parent {
  // 父类静态变量
  static int m = 1;  
  // 父类非静态变量
  int n = 1;
 
  // 静态语句块1
  static {
    m++;
    // j++; 父类非静态变量不能在静态语句块中使用
    System.out.println("父类静态初始化块一|" + m);
  }
  // 静态语句块2
  static {
    m++;
    System.out.println("父类静态初始化块二|" + m);
  }
 
  // 构造函数
  public Parent() {
    m++;
    n++;
    System.out.println("父类构造方法|" + m + "|" + n);
  }
 
  // 非静态语句块
  {
    m++;
    n++;
    System.out.println("父类非静态初始化块一|" + m + "|" + n);
  }
 
  // 非静态语句块
  {
    m++;
    n++;
    System.out.println("父类非静态初始化块二|" + m + "|" + n);
  }
 
  // 非静态方法
  public void pMethod() {
    m++;
    n++;
    System.out.println("父类非静态方法|" + m + "|" + n);
    return;
  }
 
  // 静态方法
  public static void pStaticMethod() {
    m++;
//   j++; 父类非静态变量不能在静态方法中使用
    System.out.println("父类静态方法|" + m);
    return;
  }
 
  @Override
  protected void finalize() throws Throwable {
    super.finalize();
    System.out.println("销毁父类|");
  }
}
public class Child extends Parent {
  // 静态变量
  static int i = 1;
  // 非静态变量
  int j = 1;
 
  // 静态语句块1
  static {
    m++;
    i++;
    // j++; 非静态变量不能在静态语句块中使用
    System.out.println("子类静态初始化块一 " + "|" + m + "||" + i);
  }
  // 静态语句块2
  static {
    m++;
    i++;
    System.out.println("子类静态初始化块二 " + "|" + m + "||" + i);
  }
 
  // 构造函数
  public Child() {
    m++;
    n++;
    i++;
    j++;
    System.out.println("子类构造方法 " + "|" + m + "|" + n + "|" + i + "|" + j);
  }
 
  // 非静态语句块
  {
    m++;
    n++;
    i++;
    j++;
    System.out.println("子类非静态初始化块一" + "|" + m + "|" + n + "|" + i + "|" + j);
  }
  // 非静态语句块
  {
    m++;
    n++;
    i++;
    j++;
    System.out.println("子类非静态初始化块二" + "|" + m + "|" + n + "|" + i + "|" + j);
  }
 
  // 非静态方法
  public void pMethod() {
    m++;
    n++;
    i++;
    j++;
    System.out.println("子类继承非静态方法" + "|" + m + "|" + n + "|" + i + "|" + j);
    return;
  }
 
  // 静态方法
  public static void pStaticMethod() {// 静态方法不能被继承
    m++;
    i++;
    // j++; 非静态变量不能在静态方法中使用
    return;
  }
 
  // 非静态方法
  public void cMethod() {
    m++;
    n++;
    i++;
    j++;
    System.out.println("子类非静态方法" + "|" + m + "|" + n + "|" + i + "|" + j);
    return;
  }
 
  // 静态方法
  public static void cStaticMethod() {
    m++;
    i++;
    // j++; 非静态变量不能在静态方法中使用
    System.out.println("子类静态方法 " + "|" + m + "||" + i);
    return;
  }
 
  @Override
  protected void finalize() throws Throwable {
    super.finalize();
    System.out.println("销毁子类|");
  }
}

总结

以上就是本文关于Java中类加载过程全面解析的全部内容,希望对大家有所帮助。如有问题可以随时留言,小编会及时回复大家的。期待您的宝贵意见。

相关文章

  • SpringBoot如何在运行时动态添加数据源

    SpringBoot如何在运行时动态添加数据源

    这篇文章主要介绍了SpringBoot如何在运行时动态添加数据源,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-10-10
  • springMarchal集成xStream的完整示例代码

    springMarchal集成xStream的完整示例代码

    这篇文章主要介绍了springMarchal集成xStream的示例代码,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-03-03
  • java代码执行字符串中的逻辑运算方法

    java代码执行字符串中的逻辑运算方法

    今天小编就为大家分享一篇java代码执行字符串中的逻辑运算方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • Java 8中日期和时间的处理方法

    Java 8中日期和时间的处理方法

    Java 8新增了LocalDate和LocalTime接口,接下来通过本文给大家介绍Java 8中日期和时间的处理方法,非常不错,感兴趣的朋友一起看下吧
    2016-08-08
  • Spring Boot统一接口返回及全局异常处理

    Spring Boot统一接口返回及全局异常处理

    这篇文章主要介绍了Spring Boot统一接口返回及全局异常处理,文章围绕主题展开相关资料,具有一定的参考价值需要的小伙伴可以参考一下
    2022-04-04
  • 详解使用spring boot admin监控spring cloud应用程序

    详解使用spring boot admin监控spring cloud应用程序

    这篇文章主要介绍了详解使用spring boot admin监控spring cloud应用程序,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • Spring-webflux访问关系型数据库实战

    Spring-webflux访问关系型数据库实战

    这篇文章主要为大家介绍了Spring-webflux访问关系型数据库实战详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-07-07
  • Java实现动态获取文件的绝对路径

    Java实现动态获取文件的绝对路径

    我们知道在 Java 中读取一些配置文件信息,是在开发中十分常用的要求。这篇文章就来和大家聊聊Java如何实现动态获取文件的绝对路径,感兴趣的可以了解一下
    2023-02-02
  • SpringBoot统一返回JSON格式实现方法详解

    SpringBoot统一返回JSON格式实现方法详解

    这篇文章主要介绍了SpringBoot统一返回JSON格式实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2023-02-02
  • java实现合并单元格的同时并导出excel示例

    java实现合并单元格的同时并导出excel示例

    这篇文章主要给大家介绍了关于java实现合并单元格的同时并导出excel的相关资料,文中先进行了简单的介绍,之后给出了详细的示例代码,相信对大家具有一定的参考价值,需要的朋友们下面来一起看看吧。
    2017-03-03

最新评论