• Java静态方法static关键字(2)


    被static关键字修饰的方法或者变量不需要依赖于对象来进行访问,只要类被加载了,就可以通过类名去进行访问。static是不允许用来修饰局部变量。

    问题来了,带有static关键字的类,执行的先后顺序是什么呢?先看几个例子。

    一、下面这段代码的输出结果是什么?

    public class Test extends Base{
        static{
            System.out.println("test static");
        }
        public Test(){
            System.out.println("test constructor");
        } 
        public static void main(String[] args) {
            new Test();
        }
    }
    class Base{
        static{
            System.out.println("base static");
        }
        public Base(){
            System.out.println("base constructor");
        }
    }
    base static
    test static
    base constructor
    test constructor
    View Code

    至于为什么是这个结果,我们先不讨论,先来想一下这段代码具体的执行过程,

    1. 在执行开始,先要寻找到main方法,因为main方法是程序的入口,但是在执行main方法之前,必须先加载Test类
    2. 在加载Test类的时候发现Test类继承自Base类,因此会转去先加载Base类,在加载Base类的时候,发现有static块,便执行了static块。
    3. 在Base类加载完成之后,便继续加载Test类,然后发现Test类中也有static块,便执行static块。在加载完所需的类之后,便开始执行main方法。
    4. 在main方法中执行new Test()的时候会先调用父类的构造器,然后再调用自身的构造器。因此,便出现了上面的输出结果。

    二、下面这段代码的输出结果是什么?

    public class Test {
        Person person = new Person("Test");
        static{
            System.out.println("test static");
        }
        public Test() {
            System.out.println("test constructor");
        } 
        public static void main(String[] args) {
            new MyClass();
        }
    }
     
    class Person{
        static{
            System.out.println("person static");
        }
        public Person(String str) {
            System.out.println("person "+str);
        }
    }
     
    class MyClass extends Test {
        Person person = new Person("MyClass");
        static{
            System.out.println("myclass static");
        }
         
        public MyClass() {
            System.out.println("myclass constructor");
        }
    }
    test static
    myclass static
    person static
    person Test
    test constructor
    person MyClass
    myclass constructor
    View Code

    类似地,我们还是来想一下这段代码的具体执行过程。

    1. 首先加载Test类,因此会执行Test类中的static块。接着执行new MyClass(),而MyClass类还没有被加载,因此需要加载MyClass类。
    2. 在加载MyClass类的时候,发现MyClass类继承自Test类,但是由于Test类已经被加载了,所以只需要加载MyClass类,那么就会执行MyClass类的中的static块。
    3. 在加载完之后,就通过构造器来生成对象。而在生成对象的时候,必须先初始化父类的成员变量,因此会执行Test中的Person person = new Person()
    4. 而Person类还没有被加载过,因此会先加载Person类并执行Person类中的static块,接着执行父类的构造器,完成了父类的初始化,然后就来初始化自身了
    5. 因此会接着执行MyClass中的Person person = new Person(),最后执行MyClass的构造器。

    三、这段代码的输出结果是什么?

    public class Test {
         
        static{
            System.out.println("test static 1");
        }
        public static void main(String[] args) {
             
        }
         
        static{
            System.out.println("test static 2");
        }
    }
    test static 1
    test static 2
    View Code

    虽然在main方法中没有任何语句,但是还是会输出,原因上面已经讲述过了。

    另外,static块可以出现类中的任何地方(只要不是方法内部,记住,任何方法内部都不行),并且执行是按照static块的顺序执行的。

    参考摘抄自https://www.cnblogs.com/dolphin0520/p/3799052.html

    ---------------------------------------

    执行顺序:

    1. main函数所在的类,若有父类,先加载父类的静态代码块和静态赋值、之后再加载子类的【只执行一次】
    2. 回到main函数,执行相应的对象初始化。调用类的构造函数前,先初始化父类中的成员变量,再是父类的构造函数,再是子类的成员变量-子类的构造函数
  • 相关阅读:
    JMeter——请求元件——配置元件——参数化——用户自定义变量
    JMeter——结合fiddler查看响应结果
    JMeter——断言——xpath Assertion
    JMeter——断言——响应断言
    JMeter——配置元件——http信息头管理器使用
    JMeter——查看结果树——html使用
    JMeter——查看结果树 ——css_jquery_tester(css选择器测试)
    JMeter——查看结果树
    JMeter——http请求默认值
    java.lang.RuntimeException: Cannot reconnect.
  • 原文地址:https://www.cnblogs.com/kunpengv5/p/8018779.html
Copyright © 2020-2023  润新知