static关键字的内存分析

通常情况下,Java把内存分为栈内存、堆内存和方法区

栈内存用来存放一些基本类型的变量和数组(数组也是一种引用类型)及对象的引用变量

堆内存主要是来放置对象的,即我们在程序中new出来的对象。

static,意味为静态的,用static修饰的变量和方法,实际上是给这些变量和方法指定了在内存中的”位置“(这个位置也叫静态区、方法区、数据区和共享区)。既然在内存中指定了位置,那么他们的 “大小”似乎就是固定的了,有了位置和大小的特征,在栈中或堆中开辟空间那就是非常的方便了。对于静态的东西,JVM在加载类时,就在內存中开辟了这些静态变量的空间(内存空间),即编译时就为这些成员变量的实例分配了空间。

下面我们来看一段代码

 1 package demo;
 2 
 3 public class StaticDemo {
 4     public static void main(String[] args) {
 5         Visitor visitor1 = new Visitor();
 6         System.out.println("count : " + visitor1.count);// 1
 7         System.out.println("visitCount : " + Visitor.visitCount);// 1
 8 
 9         Visitor visitor2 = new Visitor();
10         System.out.println("count : " + visitor2.count);// 1
11         System.out.println("visitCount : " + Visitor.visitCount);// 2
12 
13         Visitor visitor3 = new Visitor();
14         visitor3.count = 5;
15         visitor3.visitCount = 0; // 归0
16         System.out.println("count : " + visitor1.count);// 1
17         System.out.println("count : " + visitor2.count);// 1
18         System.out.println("count : " + visitor3.count);// 5
19         System.out.println("visitCount : " + visitor1.visitCount);// 0
20         System.out.println("visitCount : " + visitor2.visitCount);// 0
21         System.out.println("visitCount : " + visitor3.visitCount);// 0
22     }
23 }
24 
25 class Visitor {
26     int count;
27     static int visitCount;
28 
29     public Visitor() {
30         count++;
31         visitCount++;
32     }
33 }

从上面代码来看,用static修饰的变量,每个类的实例改变的都是同一个副本,即在内存中static修饰的变量只有一份,而对于普通的成员变量,每个实例都有各自的副本,我们用图来简单的分析StaticDemo类的代码,首先JVM把Visit类与StaticDemo类编译完然后加载到方法区,在Visit方法区中,JVM检查到有个static修饰的变量visitCount,则又开辟了一块内存(叫做静态区)来存放,此时内存情况

接着JVM会自动寻找main方法并在栈中为其开辟一个空间,再看代码第5行

这时,来了一个访问者visitor1,JVM在栈中为visitor1开辟了一块内存,并且指向堆中的Visitor内存空间,由于count与visitCount为成员变量,所以有默认初始值0,根据代码

第6行与第7行的运行结果

我们可以得到count与visitCount变量在堆内存中的值

同理9,10,11行

与5,6,7行分析类似,此时内存情况及count与visitCount变量的值

接下来来了一个visitor3,把count与visitCount都重新赋值了

由于visitor1,visitor2,visitor3在堆空间中都有各自的count变量,所有当visitor3改变了count变量的值时,visitor1与visitor2的count变量值并没有受影响,而当visitor3对visitCount重新赋值了,因为visitor1,visitor2,visitor3的visitCount都为静态区中的0x0001,所以用visitor1,visitor2,visitor3访问的visitCount为最新的值0。可见,静态变量与具体的实例无关,属于整个类,在编程中,当所有对象共享某个数据的时候,我们就可以把这个成员变量定义为静态的,如上面的visitCount。

原文地址:https://www.cnblogs.com/huangminwen/p/5951225.html