JVM内存结构图
实例变量、类变量、全局静态变量定义
实例变量
在类内、方法体外定义的变量并且无static、final修饰的变量称为实例变量。
实例变量存储在JVM内存结构的堆上。
实例变量是在类初始化时分配的,也就是在运行时数据区中。
-
类变量
在类内、方法体外定义的变量并且有static修饰的变量称为类变量,类变量和类关联在一起,随着类的加载而加载,他们成为类数据在逻辑上的一部分。
类变量被类的所有实例共享,即使没有类实例时,你也可以访问它
类变量则存储JVM内存结构的方法区。
类变量是在加载时经过二次初始化而被分配(class->加载->验证->准备->解析->初始化-进入运行时数据区),其中在准备时赋零值(这个也就是为啥我们不赋值也能访问不报错的原因了),在初始化时赋开发填写的值。
咱们看JVM内存结构图,类变量是在类加载子系统中分配的,而类加载子系统又分为一下几个步骤。
全局常量
在类内、方法体外定义的变量并且有static和final修饰的变量称为全局常量。
全局常量是在编译的时候就会被分配了。
咱们看JVM内存结构图,全局常量则是在Java文件编译成Class文件时就已经有了
2、代码表现
/**
* @author liuchao
* @date 2023/2/27
*/
public class Goods {
/**
* 实例变量
*/
private Long id;
/**
* 类变量
*/
public static String desc;
/**
* 全局常量
*/
public final static String goodsName;
static {
goodsName = "测试商品";
}
}
3、实际使用填坑
-
实例变量
/**
* @author liuchao
* @date 2023/2/27
*/
public class Goods {
private Long id;
public static void main(String[] args) {
/**
* 实例变量必须初始化对象后才可以正常使用,否则报错
*/
Goods g = new Goods();
System.out.println(g.id);
}
}
-
类变量
注意:仔细看代码,感觉90%的开发人员见到此面试题会答错吧
/**
* @author liuchao
* @date 2023/2/27
*/
public class Goods {
private static Long id = 20L;
static {
id = 30L;
goodsName = "商品2";
}
private static String goodsName = "商品1";
public static void main(String[] args) {
/**
* 类变量,对象无需实例化也可以访问,及时我给对象赋值null,也可正常访问
* 注意此场景,你真的知道为啥吗
*/
Goods g = null;
System.out.println(g.id);
System.out.println(Goods.id);
/**
* 注意此场景
* 因为在类加载-初始化阶段时,解析字节码是按照编码先后顺序执行的
* 因为静态代码块在类变量声明之前,所以最后执行的结果是"商品1"
*/
System.out.println(Goods.goodsName);
}
}
-
全局常量
/**
* @author liuchao
* @date 2023/2/27
*/
public class Goods {
public static final String str1 = "字符串1";
public static final String str2;
static {
str2 = "字符串2";
}
public Goods() {
/**
* 此处是不允许给全局静态变量赋值的,因为全局静态变量是在编译器就分配了,构造函数是在类实例化时调用
*/
//str2 = "xx";
}
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/144633.html