[javase学习笔记]-8.6 静态的内存载入
之前几节我们一直在说statickeyword所修饰的静态,那么在程序执行时,内存中static是怎样体现的呢?这一节我们就来看一看。
我们还是先看一个样例。希望我们通过对这个样例的分析让我们刚開始学习的人们对static所修饰的静态在内存中的详细体现有一个深刻的理解。
class Person { String name;//姓名。这是一个成员变量 int age; static String country = "美国";//国籍。这是一个静态变量 Person(String name,int age)//构造函数 { this.name = name; this.age = age; } public void printInfo()//非静态函数 { System.out.println(name+":"+age); } public static void printCoun()//静态函数,打印静态变量country { System.out.println(Person.country); } } class StaticTest { public static void main(String[] args) { Person.printCoun(); Person p = new Person("科比",37); p.printInfo(); } }我们先来看执行结果,然后再一步步分析:
样例非常easy,结果也非常明显,我们之前在7.9中谈成员变量与静态变量的差别时提到了一个差别就是成员变量是存储在堆内存中的对象中。而静态变量则存储在方法区中的静态区中。
这里,我们就引入了内存的一个新区域,那就是方法区。对于方法。当程序执行时。都会被存储在这个区域。
那么我们就对上面的代码执行过程和内存变化进行分析,当然在我们分析之前。我们必须明白一个常识。那就是当我们执行类时,类就会进入内存。
那么对于上面的代码,分析过程就会非常清晰了:
1.当我们执行程序时。StaticTest类进入内存,虚拟机会在方法区的非静态区中分配空间存储StaticTest(){}默认构造函数,同一时候在方法区的静态中分配空间存储static main(){……}主函数,当然包含主函数的全部代码的字节码。
2.静态区的main函数进栈内存,main方法中有一个对象变量p。
3.执行Person.printCoun(){}方法,Person类进入内存。方法区的非表态区分配空间存放构造函数Person(name,age){……}和非静态函数void printInfo(){……},在方法区的静态区中分配空间存储静态变量country="美国"和静态方法printCoun(){……}。
4.静态区的printCoun()方法进栈内存,并从静态区找到静态变量country并打印。控制台输出:“美国”。
5.printCoun()方法执行结束,跳出方法。printCoun()方法出栈内存。
6.执行Person p = new Person("科比",37),此时堆内存中创建空间存储对象,这里如果地址为0x0056,则所属this=0x0056,并有成员变量name和age。
7.非静态区的构造函数Person(name,age)进栈内存,对对象进行初始化,为堆内存中的对象进行初始化,name=科比,age=37。
8.初始化完毕,把地址0x0056赋值给对象p。p=0x0056。
9.构造函数出栈内存,释放參数name和age。
10.执行p.printInfo()语句,非表态区的printInfo()方法进栈内存,this=0x0056。
11.打印this所指向的成员变量this.name和this.age。控制台输出:科比:37。
12.printInfo()方法执行结束,跳出方法,方法出栈内存。
13.main()函数执行结束。跳出,函数出栈内存。
14.程序执行结束。
上面我们对样例中的代码进行了逐步分析。基本上明晰了statickeyword所修饰的静态在程序执行时在内存中的详细变化,希望在以后的实际开发过程中有所帮助。
最后我们再说一个小知识点:存储在方法区中的变量和方法都会对象所共享,所以方法区又称为共享区。