点是静态成员变量存放在方法区。JDK8之前,静态成员变量确实存放在方法区;但JDK8之后就取消了“永久代”,取而代之的是“元空间”,永久代中的数据也进行了迁移,静态成员变量迁移到了堆中(方法区是JVM的规范,永久代是方法区的具体实现)。
下面简单介绍一下其他区域:
程序计数器:又称PC寄存器,英文全称为Program Counter Register。在实际硬件平台上,PC寄存器存放下一条将要执行的指令的地址,CPU根据PC寄存器的内容找到该指令并将其放到指令寄存器中;同时,PC寄存器的中的地址加1,具体加几个字节要看机器的字长,或者由转移指令给出下一条指令的地址。指令执行完后,重复上述步骤,取指令、修改PC寄存器内容、执行指令。此处的程序计数器是JVM内部的虚拟寄存器,功能与实际密码器类似;速度特别快;每个线程都有自己的程序计数器。
堆:由所有线程共享;运行时动态申请的内存都在堆上分配,包括new的对象和数组;JDK8后,静态成员变量、常量池等也在堆中;与C不同,动态申请的内存不需要程序员回收,Java有自动垃圾回收机制。
虚拟机栈:每个线程拥有独立的栈;存放局部变量、对象引用、操作数栈、方法出口等;后进先出,被调方法结束后,对应栈区变量等立即销毁。
本地方法栈:主要与虚拟机用到native方法有关。待了解。
方法区:JDK8之前,由永久代实现,主要存放类的信息、常量池、方法数据、方法代码等;JDK8之后,取消了永久代,提出了元空间,并且常量池、静态成员变量等迁移到了堆中;元空间不在虚拟机内存中,而是放在本地内存中。那么,方法区是不是就不属于虚拟机内存的一部分了?还是元空间只是方法区的一部分,还有一部分东西存放在方法区中?待了解。
总结:栈中存在的都是线程独立的变量,
多线程的共享变量:static修饰的成员静态变量,不同Thread用的是同一个runnable对象
eg:Ticket t = new Ticket();
new Thread(t).start();
new Thread(t).start();
}
}
class Ticket implements Runnable{
这里的对象t就是线程共用的。