[javase学习笔记]-7.13 静态的内存加载

之前几节我们一直在说static关键字所修饰的静态,那么在程序运行时,内存中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();
	}
}

我们先来看运行结果,然后再一步步分析:

例子很简单,结果也很明显,我们之前在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.程序运行结束。

上面我们对例子中的代码进行了逐步分析,基本上明晰了static关键字所修饰的静态在程序运行时在内存中的具体变化,希望在以后的实际开发过程中有所帮助。

最后我们再说一个小知识点:存储在方法区中的变量和方法都会对象所共享,所以方法区又称为共享区。

时间: 2024-12-12 16:41:58

[javase学习笔记]-7.13 静态的内存加载的相关文章

[javase学习笔记]-7.4 构造函数的内存加载

这一节我们来说说构造函数在内存中是如何被加载的. 我们之前说过,构造函数只有创建对象时才会被调用并且只调用一次.那么在创建对象的过程中内存中的变化是什么样的呢? 我们接着上一节的Person类,我们分析 class Person { private String name; private int age; Person()//构造函数,分别为成员变量name和age赋固定的值 { name = "baby"; age = 1; System.out.println("per

NGUI学习笔记(四):动态加载UI和NGUI事件

动态加载UI 我们进入一个场景后,如果将这个场景所有可能用到的UI都直接放在场景中做好,由于要在进入场景时就部署好所有的UI对象,那么当UI对象较多时会碰到的问题是:1.初始化场景会产生非常明显的卡顿.2.所有UI都在场景中导致占用大量的内存. 所以我们需要对UI组件进行动态加载和销毁,当需要打开指定的UI时,动态的创建出这个UI对象,而当关闭这个UI对象之后,可以对其进行销毁从而释放出内存. 将UI制作成一个预制件 我们可以在场景中制作好一个UI,然后将其保存成一个预制件后从场景中移除,然后我

[javase学习笔记]-8.7 静态代码块

这一节我们看一个比較特殊的概念,那就是静态代码块. 前面我们也提到过代码块,就是一段独立的代码空间.那么什么是静态代码块呢?说白了,就是用statickeyword修饰的代码块. 我们来看一个样例: class StaticBlock { static { System.out.println("静态代码块被运行"); } void myPrint() { System.out.println("myPrint方法运行"); } } class StaticBloc

【extjs6学习笔记】1.7 初始:加载第三方库

https://www.sencha.com/blog/integrating-ext-js-with-3rd-party-libraries-2/ Introduction Ext JS provides a lot of built-in components right out of the box that are highly customizable. If it's not in the framework, you can easily extend the classes or

【数据库学习笔记】(2)JDBC加载并注册数据库驱动并连接数据库

Android学习笔记—第六章 Asynctask异步加载

第六章 Asynctask 异步加载 1.好处:不需要创建线程就可管理线程 缺点:步骤多 2.步骤: (1)创建一个类继承Asynctask<xxx,xxx,xxx>; 三个泛型参数: 第一个:决定了execute()方法的传入值类型,决定了doInBackground()方法的传入值类型 第二个:决定了publishProgress()方法的传入值类型,决定了onProgressUpdate()方法的传入值类型 第三个:决定了doInBackground()方法的返回值类型,决定了onPos

【Qt学习笔记】13.拖放技术:Drag & Drop

1.接受拖放 Drag & Drop 是一个界面操作,用于在两个窗口间传递数据. Drag Source: 拖放源窗口 Drag Target: 拖放目标窗口 拖放操作: 1.在源窗口:选中目标,按下鼠标,移动,拖至目标窗口(Drag) 2.在目标窗口:取消鼠标,到指定位置,松开鼠标(Drop) (按下ESC取消操作) MIME: MIME(Multipurpose Internet Mail Extensions)被传递的数据以MIME格式传送,它是多组type-data数据:(type0,

javasE学习笔记:关键字super的使用

/* super 的作用: 1 . super可以在子类中引用父类的成员,通过 .的方式和属性.this相对应. 2. 在子类的构造方法中可以可使用super(参数列表)语句调用父类的构造方法 3. 注意: 子类的构造方法中一定要调用父类的构造方法.... */ javasE学习笔记:关键字super的使用,布布扣,bubuko.com

打造DLL内存加载引擎学习笔记

原文:http://www.pediy.com/kssd/index.html -- 病毒技术 -- 病毒知识 -- Anti Virus专题 首先看下我们内存加载引擎的流程. 1. 申请一段大小为dll映射内存后的映像大小的内存空间. 2. 移动各个区段的数据到申请的内存. 2. 修复引入表结构的地址表. 4. 通过重定位结构修复需要重定位的地址. 5. 调用DllMain入口点 流程解析: pe结构中nt header结构当中的ImageSize存放的是我们整个文件映射到内存后的映像大小,这