[javase学习笔记]-8.6 静态的内存载入
2018-04-19 18:14
190 查看
之前几节我们一直在说statickeyword所修饰的静态,那么在程序执行时,内存中static是怎样体现的呢?这一节我们就来看一看。
我们还是先看一个样例。希望我们通过对这个样例的分析让我们刚開始学习的人们对static所修饰的静态在内存中的详细体现有一个深刻的理解。
样例非常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所修饰的静态在程序执行时在内存中的详细变化,希望在以后的实际开发过程中有所帮助。
最后我们再说一个小知识点:存储在方法区中的变量和方法都会对象所共享,所以方法区又称为共享区。
我们还是先看一个样例。希望我们通过对这个样例的分析让我们刚開始学习的人们对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所修饰的静态在程序执行时在内存中的详细变化,希望在以后的实际开发过程中有所帮助。
最后我们再说一个小知识点:存储在方法区中的变量和方法都会对象所共享,所以方法区又称为共享区。
相关文章推荐
- [javase学习笔记]-7.13 静态的内存加载
- [javase学习笔记]-4.3 函数的内存加载过程
- 【JavaSE学习笔记】IO流06_其他常用流(基本数据类型流、内存操作流、打印流、标准流、随机访问流、合并流、序列化与反序列化流)、属性集合类Properties
- C和指针之动态内存分配堆、栈、全局区(静态区)、常量区对比总结学习笔记
- linux内存学习笔记(三)——静态映射与DMA
- [javase学习笔记]-7.4 构造函数的内存加载
- [javase学习笔记]-5.2 数组的内存分配及特点
- [javase学习笔记]-8.7 静态代码块
- [javase学习笔记]-6.3 对象的内存体现
- [javase学习笔记]-8.7 静态代码块
- <JavaSE学习笔记>面向对象(2):Java内存机制
- [javase学习笔记]-6.3 对象的内存体现
- C++学习笔记3:内存模型和名称空间
- Javascript学习笔记3——变量、作用域和内存问题
- [置顶] JavaSE学习笔记_8:Java异常
- 黑马程序员————IOS学习笔记 第7篇 内存管理
- JAVA学习笔记--4.多线程编程 part1.背景知识和内存模型
- (原创)c#学习笔记08--面向对象编程简介01--面向对象编程的含义03--静态和实例类成员
- 黑马程序员--IOS_学习笔记_内存管理相关@porperty
- windows api学习笔记-读写其他进程的内存