您的一个赞或者是评论区的一句话都将是对我最大的激励。
刚学编程的时候,被什么引用变量,静态变量,对象搞得混淆不清,给自己学习编程增加了很大的难度,希望通过本篇文章,能对大家有所帮助。
先来介绍一个简单的概念:
Case cc =new Case(); //这是我们很熟悉的语句,这条语句可以分解为 Case cc; //其中cc叫做引用变量,在栈内存里面开辟,此时cc=null cc=new Case(); // new Case()是在堆内存里面开辟了空间给Case类的对象,这个对象没有名字, //Case()随即调用了Case类的构造函数,把对象的地址在堆内存的地址给引用变量cc
我们不要把事情想的太复杂了,简单明了的说 Java虚拟机的内存可以分为三个区域:栈stack、堆heap、方法区method area
这3个区各有各的特点,我简单整理罗列了一下方便各位理解:
栈的特点如下:
1. 栈描述的是方法执行的内存模型。每个方法被调用都会创建一个栈帧(存储局部变量、操作数、方法出口等)
2. JVM为每个线程创建一个栈,用于存放该线程执行方法的信息(实际参数、局部变量等)
3. 栈属于线程私有,不能实现线程间的共享!
4. 栈的存储特性是“先进后出,后进先出”
5. 栈是由系统自动分配,速度快!栈是一个连续的内存空间!
堆的特点如下:
1. 堆用于存储创建好的对象和数组(数组也是对象)
2. JVM只有一个堆,被所有线程共享
3. 堆是一个不连续的内存空间,分配灵活,速度慢!
方法区(又叫静态区)特点如下:
1. JVM只有一个方法区,被所有线程共享!
2. 方法区实际也是堆,只是用于存储类、常量相关的信息!
3. 用来存放程序中永远是不变或唯一的内容。(类信息Class对象、静态变量、字符串常量等)
方法区也是位于堆中的,只不过作用较为特殊,我们单独提取出来。所以java虚拟机内存其实就只有栈 和 堆 ,下面我放一张图让大家进一步理解2这的区别。
为了让老年人也看得懂,我再放一张图,简单描述三者在内存中的关系
结合我最先前说的Case cc =new Case()的概念,这样我们就明确了
这里的“=”并不是赋值的意思,而是把对象的地址传递给变量,有点类似指针。
1.图中的15db9742和15db9753这个数字是我自己编的,代表对象在堆中的地址,但是这个地址是会变化的,这个要特别注意!
2.字符串常量一般也都是存储在方法区中
懂了之后我们来看三个问题
题1:
int a = 1 ,a和1分别存在于哪里?
答:
1. a作为类的成员变量,存放于方法区中;1保存在堆(Heap)的实例中
2. a作为方法局部变量,存放于Java虚拟机栈(JVM Stacks)的局部变量表中;1保存在栈内存中。
题2:
String str=new String(“abc”) 创建了几个对象
答:
一个或者2个, 我们可以把上面这行代码分成 String str 、= 、 “abc” 和 new String() 四部分来看待。 String str只是定义了一个名 为str的String类型的变量,因此它并没有创建对象;
= 号是对变量str进行初始化,将某个对象的引用赋值给它,也没有创建对象;
所以说如果字符串常量池里有"abc",我们就只创建了 new String一个对象;
如果没有"abc" ,我们就创建了 new String和 "abc"2个对象。
题3:
String a=“123” a在哪里?“123”又存储在哪里
答:
通过题1,我们已经有经验了,如果说a是成员变量,则a存在方法区中,如果说a是局部变量,则a存在栈中,“123”则作为字符串变量存在方法区中。
本篇文章我还是介绍的比较详细的,相信大家都有所收获。
白嫖不好,创作不易。各位的点赞就是我创作的最大动力,如果我有哪里写的不对,欢迎评论区留言进行指正,我们下篇文章见!