类型信息之类字面常量

类是程序的一部分,每个类都有一个Class对象,换言之,每当编写并且编译一个新类,就会产生一个Class对象。
为了使用类而做的工作实际包含三个步骤:
1.加载,这是由类加载器执行的。改步骤将查找字节码,并从字节码中创建一个Class对象。

2.链接。在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。

3.初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。

下面通过几个例子来说明


class P{
static int a = 47;
static{
System.out.println("asdasdasd");
}
}

public class StaticTest {
public static void aa(){
System.out.println("/.................");
}

public static void main(String[] args) throws ClassNotFoundException {
// Class c = Class.forName("P");
System.out.println(P.a);
}
}

当main方法中执行注释或者输出语句时都会执行类P中的静态域和静态块里面的语句。
如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接(为这个域分配存储空间)和初始化(初始化存储空间)。

下面看另外一个例子

package com.unis.type;

import java.util.Random;
class Initable{
static final int staticFinal =47;
static final int staticFinal2= ClassInitialization.rand.nextInt(1000);
static{
System.out.println("Initializing Initable");
}

public Initable() {
System.out.println("asdsadasdasdsssssssssssssssssssssssssss");
}
}

public class ClassInitialization {
static{
System.out.println("ClassInitialization");
}
public static Random rand = new Random(47);
public static void main(String[] args) throws ClassNotFoundException {
System.out.println(Initable.staticFinal);
}
}

上面这个例子会输出ClassInitialization 和47,输出ClassInitialization很容易理解,是因为要执行main方法,所以要对类ClassInitialization进行初始化。
可是为什么没有输出静态块里面的内容呢(Initializing Initable)?,这是因为静态域staticFinal是final的,如果一个静态域是final的,那么这个值就是“编译器常量”,就像Initable.staticFinal那样,那么这个值不需要对Initable类进行初始化就可以读取。但是,如果只是将一个域设置为static和final的,还不足以确保这种行为。例如下面的例子

package com.unis.type;

import java.util.Random;
class Initable{
static final int staticFinal =47;
static final int staticFinal2= ClassInitialization.rand.nextInt(1000);
static{
System.out.println("Initializing Initable");
}

public Initable() {
System.out.println("asdsadasdasdsssssssssssssssssssssssssss");
}
}

public class ClassInitialization {
static{
System.out.println("ClassInitialization");
}
public static Random rand = new Random(47);
public static void main(String[] args) throws ClassNotFoundException {
System.out.println(Initable.staticFinal2);
}
}

这个例子中staticFinal2同样是static和final的,当执行Initable.staticFinal2时则会输出静态块中的Initializing Initable。对Initable.staticFinal2的访问将强制进行类的初始话,因为它不是一个编译器常量。

下面看另外一个例子

package com.unis.type;

import java.util.Random;
class Initable{
static final int staticFinal =47;
static final int staticFinal2= ClassInitialization.rand.nextInt(1000);
static{
System.out.println("Initializing Initable");
}

public Initable() {
System.out.println("asdsadasdasdsssssssssssssssssssssssssss");
}
}

public class ClassInitialization {
static{
System.out.println("ClassInitialization");
}
public static Random rand = new Random(47);
public static void main(String[] args) throws ClassNotFoundException {
Class initable=Initable.class;
}


执行这段代码发现控制台什么都不会输出,静态代码块中的代码没有被执行。
而在第一个例子中Class c = Class.forName("P");同样是返回一个Class对象,第一个例子则会执行静态块中的内容。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值