Java类的初始化顺序

简介: 做Java开发的,起码要知道类的初始化顺序,因为这样你才知道编写的代码运行到了那里。与文无关一段代码public class BaseTest { private String baseName = "baseName"; static { System.

做Java开发的,起码要知道类的初始化顺序,因为这样你才知道编写的代码运行到了那里。

img_2d9f75e7088238efb0ff94c2b9653e45.png
与文无关

一段代码

public class BaseTest {
    private String baseName = "baseName";


    static {
        System.out.println("父类静态代码块");
    }



    public BaseTest() {
        System.out.println("父类构造方法");
        callName();
    }

    public void callName(){
        System.out.println(baseName);
    }

    static class Sub extends BaseTest{
        private String baseName = "subName";

        public Sub() {
            System.out.println("子类构造方法");
        }

        static {
            System.out.println("子类静态代码块");
        }

        @Override
        public void callName() {
            System.out.println(baseName);   //输出为null
            super.callName();   //输出baseName,baseStatic
        }

    }

    public static void main(String[] args) {
        Sub sub = new Sub();
    }

}


以上代码运行的结果如下:

父类静态代码块
子类静态代码块
父类构造方法
null
baseName
子类构造方法

为什么在子类中直接调用 System.out.println(baseName),输出为null呢?
在创造派生类的过程中首先创建基类对象,然后才能创建派生类。创建基类即默认调用BaseTest()方法,在方法中调用callName()方法,由于派生类中存在此方法,则被调用的callName()方法是派生类中的方法,此时派生类还未构造,所以变量baseName的值为null

静态代码块的初始化

public class StaticTest {

    static {
        System.out.println("父类的静态代码块");
    }

    public static String str = getStr();

    private static String getStr() {
        System.out.println("父类的静态方法");
        return "str1";
    }

    public StaticTest() {
        System.out.println("父类构造方法");
    }

    static class SubStatic extends StaticTest{
        static {
            System.out.println("子类的静态代码块");
        }

        public static String str1 = getStr1();

        private static String getStr1() {
            System.out.println("子类的静态方法");
            return "str1";
        }

        public SubStatic() {
            System.out.println("子类构造方法");
        }
    }

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

运行结果:

父类的静态代码块
父类的静态方法
子类的静态代码块
子类的静态方法
父类构造方法
子类构造方法

最后

动手操作,查看了下类的初始化过程。

相关文章
|
2月前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
164 57
|
14天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
2月前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
66 8
|
2月前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
79 17
|
2月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
2月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
140 4
|
2月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
87 2
|
2月前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
74 4
|
2月前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
56 5
|
2月前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
123 5