当前位置: 首页 > news >正文

java类初始化顺序

转自:http://zangweiren.iteye.com/blog/208122

对于静态变量、静态初始化块、变量、初始化块、构造器,它们的初始化顺序以此是(静态变量、静态初始化块)>(变量、初始化块)>构造器。我们也可以通过下面的测试代码来验证这一点: Java代码
    public class InitialOrderTest {

        // 静态变量
        public static String staticField = "静态变量";
        // 变量
        public String field = "变量";

        // 静态初始化块
        static {
            System.out.println(staticField);
            System.out.println("静态初始化块");
        }

        // 初始化块
        {
            System.out.println(field);
            System.out.println("初始化块");
        }

        // 构造器
        public InitialOrderTest() {
            System.out.println("构造器");
        }

        public static void main(String[] args) {
            new InitialOrderTest();
        }
    }


    运行以上代码,我们会得到如下的输出结果:
    静态变量
    静态初始化块
    变量
    初始化块
    构造器
    这与上文中说的完全符合。

 

那么对于继承情况下又会怎样呢?我们仍然以一段测试代码来获取最终结果: Java代码
    class Parent {
        // 静态变量
        public static String p_StaticField = "父类--静态变量";
        // 变量
        public String p_Field = "父类--变量";

        // 静态初始化块
        static {
            System.out.println(p_StaticField);
            System.out.println("父类--静态初始化块");
        }

        // 初始化块
        {
            System.out.println(p_Field);
            System.out.println("父类--初始化块");
        }

        // 构造器
        public Parent() {
            System.out.println("父类--构造器");
        }
    }

    public class SubClass extends Parent {
        // 静态变量
        public static String s_StaticField = "子类--静态变量";
        // 变量
        public String s_Field = "子类--变量";
        // 静态初始化块
        static {
            System.out.println(s_StaticField);
            System.out.println("子类--静态初始化块");
        }
// 初始化块
        {
            System.out.println(s_Field);
            System.out.println("子类--初始化块");
        }

        // 构造器
        public SubClass() {
            System.out.println("子类--构造器");
        }

        // 程序入口
        public static void main(String[] args) {
            new SubClass();
        }
    }


     运行一下上面的代码,结果马上呈现在我们的眼前:
    父类--静态变量
    父类--静态初始化块
    子类--静态变量
    子类--静态初始化块
    父类--变量
    父类--初始化块
    父类--构造器
    子类--变量
    子类--初始化块
    子类--构造器
    现在,结果已经不言自明了。大家可能会注意到一点,那就是,并不是父类完全初始化完毕后才进行子类的初始化,实际上子类的静态变量和静态初始化块的初始化是在父类的变量、初始化块和构造器初始化之前就完成了。 那么对于静态变量和静态初始化块之间、变量和初始化块之间的先后顺序又是怎样呢?是否静态变量总是先于静态初始化块,变量总是先于初始化块就被初始化了呢?实际上这取决于它们在类中出现的先后顺序。我们以静态变量和静态初始化块为例来进行说明。 同样,我们还是写一个类来进行测试: Java代码
    public class TestOrder {
        // 静态变量
        public static TestA a = new TestA();

        // 静态初始化块
        static {
            System.out.println("静态初始化块");
        }

        // 静态变量
        public static TestB b = new TestB();

        public static void main(String[] args) {
            new TestOrder();
        }
    }

    class TestA {
        public TestA() {
            System.out.println("Test--A");
        }
    }

    class TestB {
        public TestB() {
            System.out.println("Test--B");
        }
    }


     运行上面的代码,会得到如下的结果:
    Test--A
    静态初始化块
    Test--B
    大家可以随意改变变量a、变量b以及静态初始化块的前后位置,就会发现输出结果随着它们在类中出现的前后顺序而改变,这就说明静态变量和静态初始化块是依照他们在类中的定义顺序进行初始化的。同样,变量和初始化块也遵循这个规律。 了解了继承情况下类的初始化顺序之后,如何判断最终输出结果就迎刃而解了。

相关文章:

  • bootstrap总结
  • java创建对象的四种方式
  • java基础之String
  • 为什么单例对象的并发调用需要同步?
  • Spring_事务(1)
  • java集合框架总结
  • LeetCode-Count Bits
  • Java中对HashMap的深度分析与比较
  • java 线程小结
  • JAVA中精确计算金额BigDecimal
  • java并发编程实践笔记
  • 第四章 進程調度
  • volatile原理与技巧
  • java I/O的基本使用
  • java多线程中的join方法详解
  • 【前端学习】-粗谈选择器
  • 【知识碎片】第三方登录弹窗效果
  • iOS筛选菜单、分段选择器、导航栏、悬浮窗、转场动画、启动视频等源码
  • JavaScript实现分页效果
  • Java深入 - 深入理解Java集合
  • mockjs让前端开发独立于后端
  • Mysql数据库的条件查询语句
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • ReactNativeweexDeviceOne对比
  • TCP拥塞控制
  • Twitter赢在开放,三年创造奇迹
  • vue:响应原理
  • 百度小程序遇到的问题
  • 不上全站https的网站你们就等着被恶心死吧
  • 代理模式
  • 搞机器学习要哪些技能
  • 前端技术周刊 2018-12-10:前端自动化测试
  • 入门级的git使用指北
  • 试着探索高并发下的系统架构面貌
  • 小试R空间处理新库sf
  • 移动端解决方案学习记录
  • 用jquery写贪吃蛇
  • Semaphore
  • ​​​​​​​GitLab 之 GitLab-Runner 安装,配置与问题汇总
  • ​iOS实时查看App运行日志
  • ​LeetCode解法汇总2670. 找出不同元素数目差数组
  • # 深度解析 Socket 与 WebSocket:原理、区别与应用
  • #HarmonyOS:软件安装window和mac预览Hello World
  • (02)vite环境变量配置
  • (10)Linux冯诺依曼结构操作系统的再次理解
  • (C语言)二分查找 超详细
  • (Ruby)Ubuntu12.04安装Rails环境
  • (超简单)构建高可用网络应用:使用Nginx进行负载均衡与健康检查
  • (附源码)计算机毕业设计ssm电影分享网站
  • (黑马出品_高级篇_01)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式
  • (一)ClickHouse 中的 `MaterializedMySQL` 数据库引擎的使用方法、设置、特性和限制。
  • (一)RocketMQ初步认识
  • (原創) 如何將struct塞進vector? (C/C++) (STL)
  • (转)setTimeout 和 setInterval 的区别
  • **CI中自动类加载的用法总结