android堆区栈区方法区,java 堆(heap)、栈(stack)和方法区(method)

JAVA的JVM的内存可分为3个区:堆(heap)、栈(stack)和方法区(method)

堆区:

1.存储的是new出来的对象和数组,每一个对象都包含一个与之对应的class的信息。(class的目的是获得操做指令)

2.jvm只有一个堆区(heap)被全部线程共享,堆中不存放基本类型和对象引用,只存放对象自己

栈区:

1.每一个线程包含一个栈区,栈中保存的是全部的变量,包括基本类型和引用类型,栈中的每一个变量都包含类型、名称、值这些内容,只不过基本类型变量的值为一个具体的值,而引用类型的变量的值为对象在堆中的地址。

2.每一个栈中的数据(原始类型和对象引用)都是私有的,其余栈不能访问。

3.栈分为3个部分:基本类型变量区、执行环境上下文、操做指令区(存放操做指令)。

方法区:

1.又叫静态区,跟堆同样,被全部的线程共享。方法区包含全部的class和static变量。java

2.方法区中包含的都是在整个程序中永远惟一的元素,如class,static变量。数组

3.字符串常量池就是存放在方法区。(具体缘由参见:http://zangxt.iteye.com/blog/472236)jvm

堆和栈的不一样:测试

<1>存数数据不一样spa

<2>回收方式不一样线程

栈中当超过变量的做用域后,java会自动释放掉为该变量分配的内存空间,该内存空间能够马上被另做他用。对象

数组和对象在没有引用变量指向它的时候,才变成垃圾,不能再被使用,可是仍然占着内存,在随后的一个不肯定的时间被垃圾回收器释放掉。blog

<3>速度不一样进程

堆中分配的内存,由Java虚拟机的自动垃圾回收器来管理。内存

堆和栈的优缺点:

<1>堆的优点是能够动态地分配内存大小,生存期也没必要事先告诉编译器,由于它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些再也不使用的数据。但缺点是,因为要在运行时动态分配内存,存取速度较慢.

<2>栈的优点是,存取速度比堆要快,仅次于寄存器,栈数据能够共享。但缺点是,存在栈中的数据大小与生存期必须是肯定的,缺少灵活性

举例: int a = 3;  int b = 3;

编译器先处理int a = 3;首先它会在栈中建立一个变量为a的引用,而后查找栈中是否有3这个值,若是没找到,就将3存放进来,而后将a指向3。

接着处理int b = 3;在建立完b的引用变量后,由于在栈中已经有3这个值,便将b直接指向3。这样,就出现了a与b同时均指向3的状况。

这时,若是再令a=4;那么编译器会从新搜索栈中是否有4值,若是没有,则将4存放进来,并令a指向4;若是已经有了,则直接将a指向这个地址。

所以a值的改变不会影响到b的值。

简单代码语句的执行过程:

dccbdf06c282bb4eafdc42437a37b380.png

系统收到了咱们发出的指令,启动了一个Java虚拟机进程,这个进程首先从classpath中找到AppMain.class文件,读取这个文件中的二进制数据,

而后把Appmain类的类信息存放到运行时数据区的方法区中。这一过程称为AppMain类的加载过程。

接着,Java虚拟机定位到方法区中AppMain类的Main()方法的字节码,开始执行它的指令。这个main()方法的第一条语句就是:

Sample test1=new Sample("测试1");

就是让java虚拟机建立一个Sample实例,而且呢,使引用变量test1引用这个实例。就让咱们来跟踪一下Java虚拟机,看看它到底是怎么来执行这个任务的:

一、 Java虚拟机直奔方法区,先找到Sample类的类型信息。结果这会儿的方法区里尚未Sample类。因而立马加载了Sample类,把Sample类的类型信息存放在方法区里。

二、 为一个新的Sample实例分配内存, 这个Sample实例持有着指向方法区的Sample类的类型信息的引用。而这个引用地址,就存放了在Sample实例的数据区里。

三、 在JAVA虚拟机进程中,每一个线程都会拥有一个方法调用栈,用来跟踪线程运行中一系列的方法调用过程,栈中的每个元素就被称为栈帧,每当线程调用一个方法的时候就会向方法栈压入一个新帧。这里的帧用来存储方法的参数、局部变量和运算过程当中的临时数据。OK,原理讲完了,就让咱们来继续咱们的跟踪行动!位于“=”前的Test1是一个在main()方法中定义的变量,可见,它是一个局部变量,所以,它被会添加到了执行main()方法的主线程的JAVA方法调用栈中。而“=”将把这个test1变量指向堆区中的Sample实例,也就是说,它持有指向Sample实例的引用。

接下来,JAVA虚拟机将继续执行后续指令,在堆区里继续建立另外一个Sample实例,而后依次执行它们的printName()方法。当JAVA虚拟机执行test1.printName()方法时,JAVA虚拟机根据局部变量test1持有的引用,定位到堆区中的Sample实例,再根据Sample实例持有的引用,定位到方法去中Sample类的类型信息,从而得到printName()方法的字节码,接着执行printName()方法包含的指令。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值