十、初始化其他情况及总结

1、猜输出
class Parent2 {
    static int a = 3;

    static {
        System.out.println("Parent2 static block");
    }
}

class Child2 extends Parent2 {
    static int b = 4;

    static {
        System.out.println("Child2 static block");
    }
}


public class MyTest10 {
    static {
        System.out.println("MyTest10 static block");
    }

    public static void main(String[] args) {
        Parent2 parent2;
        System.out.println("======");
        parent2 = new Parent2();
        System.out.println("======");
        System.out.println(parent2.a);
        System.out.println("======");
        System.out.println(Child2.b);
    }
}

我的想法:
Parent2虽然先定义了,但是new的操作是在后面,在定义的这个地方应该是不会导致初始化的,所以结果应该是:

MyTest10 static block
======
Parent2 static block
======
3
======
Child2 static block
4

实际结果:

MyTest10 static block
======
Parent2 static block
======
3
======
Child2 static block
4
2、猜输出
class Parent3 {
    static int a = 3;

    static {
        System.out.println("Parent3 static block");
    }

    static void doSomething() {
        System.out.println(" do something");
    }
}

class Child3 extends Parent3 {
    static {
        System.out.println("Child3 static block");
    }
}


public class MyTest11 {
    public static void main(String[] args) {
        System.out.println(Child3.a);
        System.out.println("======");
        Child3.doSomething();
    }
}

我的想法:
当调用Child3.a的时候,根据只有直接定义这个静态字段的类才会被初始化,所以Parent3会被初始化。但是第二行呢,调用了静态方法,Parent3肯定不会再被初始化了,但是child3会被初始化吗?不知道,因为七条中有一条:调用一个类的静态方法,但是对于静态方法却没有类似于静态字段的限制说明,那应该会被初始化吧。输出应该如下:

Parent3 static block
3
======
Child3 static block
do something

实际结果:

Parent3 static block
3
======
 do something

脸都被打肿了。
结论:
对于静态方法来说,也有跟静态字段一样的限制:只有直接定义静态方法的类才会被初始化

3、猜输出
class CL {
    static {
        System.out.println("class CL");
    }
}

public class MyTest12 {
    public static void main(String[] args) throws ClassNotFoundException {
        ClassLoader loader = ClassLoader.getSystemClassLoader();
        Class<?> aClass = loader.loadClass("com.jvm.classloader.CL");
        System.out.println(aClass);
        System.out.println("=========");
        aClass = Class.forName("com.jvm.classloader.CL");
        System.out.println(aClass);
    }
}

实际输出:

class com.jvm.classloader.CL
=========
class CL
class com.jvm.classloader.CL

分析:说明类加载的时候不会进行初始化,而反射调用的时候会进行初始化,反射也是七种情况之一。之前都没注意到。

4、猜输出
interface Parent5 {
    Thread thread = new Thread() {
        {
            System.out.println("interface thread");
        }
    };

    default int getNum() {
        return 0;
    }
}

class Child5 implements Parent5 {

    public static int a = 0;

    static {
        System.out.println("MyChild5");
    }

}


public class MyTest13 {
    public static void main(String[] args) {
        System.out.println(Child5.a);
    }
}

实际结果:

interface thread
MyChild5
0

分析:
当一个接口中定义了jdk8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,则该接口要在其之前被初始化。
这个就是原因。


注:
在接口中写默认方法时可能会遇到错误:Extension methods are not supported at language level '5',解决方法:
1、设置当前模块的 Source Language Level:
File -> Project Structure -> Modules -> Sources -> Language Level
选择 8 - Lambdas, type annotations etc.
2、设置当前模块的 Target Language Level:
File -> Settings -> File | Settings | Build, Execution, Deployment -> Compiler -> Java Compiler -> Per-module bytecode version -> Target bytecode version

5、初始化情况总结(以下要求全文背诵)

1、遇到new、getstatic、putstatic、invokestatic这四条字节码指令时,如果类型没有进行过初始化,则需要先触发其初始化阶段。能够生成这四条指令的典型java代码场景有:

  • 使用new关键字实例化对象的时候
  • 读取或设置一个类型的静态字段(被final修饰、已在编译器把结果放入常量池的静态字段除外)的时候
  • 调用一个类型的静态方法的时候
这一条有如下要注意的:
1、那四条字节码指令应该可以通过javap -c .class文件反编译之后看到。
2、对于静态字段和静态方法,只有定义这个字段或者方法的类才会被初始化。
3、运行期静态常量(即那种被排除的情况)不会被初始化
4、编译器静态常量在编译的时候,该常量就被方法调用该常量的方法所在类的常量池中了,所以不会被初始化

2、使用java.lang.reflect包的方法对类型进行反射调用的时候 ,如果类型没有进行过初始化,则需要先触发其初始化

3、当初始化的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化

4、当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类

5、当一个接口中定义了jdk8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化

6、当使用jdk7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为REF_getStatic、REF_putStatic、REF_invokeStatic、REF_newInvokeStatic四种类型的方法句柄,并且这个方法句柄对应的类没有进行过初始化,则需要先触发其初始化。


此外,还需要注意,对于一般接口,第三条不成立;且只有在调用接口的运行期常量的时候,才会导致接口的初始化。

### C语言中浮点数的初始化方法 在C语言中,浮点数可以通过多种方式进行初始化。以下是几种常见的初始化方式及其特点: #### 1. 使用直接赋值的方式 可以直接通过常量数值来初始化浮点数变量。需要注意的是,默认情况下带有小数点的数字会被认为是`double`类型,因此如果要将其存储为`float`类型,则需要显式指定。 ```c float a = 10.0f; // 显式声明为float类型 float b = 3.14F; // F或f均可用于表示float类型 ``` 上述代码片段展示了如何使用带后缀`f`或`F`的方式来确保编译器将这些字面量识别为`float`而不是`double`[^3]。 #### 2. 不带小数点的情况 如果不带小数点而只提供整数形式的数据,则该数据通常被解释为整型。在这种场景下,可以利用隐式的类型转换完成初始化过程。 ```c float c = 5; // 隐式转换成float float d = (float)7; // 显示强制转换为float ``` 这里值得注意的是,虽然能够成功运行并得到预期的结果,但从编码习惯的角度来看,建议尽量明确表达意图以提高程序可读性和减少潜在错误风险[^3]。 #### 3. 利用科学记数法进行初始化 除了普通的进制写法外,还可以借助科学记数法简化极大极小数值的表现形式。 ```c float e = 3.23e5f; // 等价于3.23 * 10^5, 后缀f表明这是单精度浮点数 float f = -4.16E-9F; // 负数同样适用此规则 ``` 按照标准化原则,这里的指数标记实际上代表乘方操作,并且支持大小写字母互换使用[^5]。 #### 4. 动态计算后的立即赋值 也可以基于其他已知条件即时演算得出目标初值后再赋予相应变量名之上。 ```c int numerator = 2; int denominator = 3; float g = ((float)numerator)/denominator; // 将分子转为float避免整除效应 ``` 在此例子当中,为了避免发生整数之间的相除而导致丢失精度的现象,提前把其中一个参与运算的操作数提升至更高的优先级即变为浮点数形态即可达成目的。 --- ### 总结 综上所述,在实际开发过程中可以根据具体需求灵活选用不同的初始化策略。无论是简单的固定值设定还是复杂的逻辑推导结果填充,只要遵循基本语法规则就能顺利完成任务。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值