Java 中存储的静态方法和静态变量在哪里?

例如:

class A {
static int i=0;
static int j;


static void method() {
// static k=0; can't use static for local variables only final is permitted
// static int L;
}
}

这些变量将存储在 Java、堆或堆栈内存中的什么地方? 它们是如何存储的?

193978 次浏览

它存储在类定义引用的堆中。仔细想想,它与堆栈没有任何关系,因为没有作用域。

静态变量存储在堆中

静态方法(实际上是所有方法)和静态变量都存储在堆的 PermGen部分中,因为它们是反射数据(与类相关的数据,而不是与实例相关的数据)的一部分。从 Java8开始,PermGen 已经被 MetaSpace 取代,而且根据 JEP 122,它只保存元数据,而静态字段存储在堆中。

请注意,这主要适用于 Oracle 的 Hotspot JVM 和其他基于它的 JVM。然而,并不是每个 JVM 都有像 Eclipse OpenJ9那样的 PermGen 或 Metaspace。

需要澄清的最新情况 :

注意,只有变量及其技术值(原语或引用)存储在 PermGen 空间中。

如果您的静态变量是对某个对象的引用,则该对象本身存储在堆的正常部分(年轻/老一代或幸存者空间)中。这些对象(除非它们是类等内部对象)是存储在 PermGen 空间中的 没有

例如:

static int i = 1; //the value 1 is stored in the PermGen section
static Object o = new SomeObject(); //the reference(pointer/memory address) is stored in the PermGen section, the object itself is not.

关于垃圾收集:

没有依赖于 finalize(),因为它不能保证运行。完全由 JVM 决定何时运行垃圾收集器以及收集什么,即使对象符合垃圾收集的条件。

当然,您可以将静态变量设置为 null,从而删除对堆上对象的引用,但这并不意味着垃圾收集器 威尔会收集它(即使没有更多的引用)。

此外,finalize()只运行一次,因此您必须确保它不会抛出异常或以其他方式阻止收集对象。如果通过某些异常停止终止,则不会在同一对象上第二次调用 finalize()

最后需要注意的是: 代码、运行时数据等的存储方式取决于所使用的 JVM,也就是说,HotSpot 的存储方式可能不同于 JRockit,甚至可能不同于同一个 JVM 的版本。上面的内容是基于 Java5和 Java6的 HotSpot (它们基本上是一样的) ,因为在回答这个问题的时候,我认为大多数人都在使用这些 JVM。由于 Java 8的内存模型发生了重大变化,上面的语句可能不适用于 Java 8 HotSpot ——而且我没有检查 Java 7 HotSpot 的变化,所以我 猜猜看上面的语句对于那个版本仍然适用,但是我在这里不确定。

这是一个回答简单、答案冗长的问题。

简单的答案是堆。类和应用于类的所有数据(不是实例数据)都存储在堆的永久生成部分中。

长远的答案已经在堆栈溢出方面:

有一个 全面描述 JVM 中的内存和垃圾收集以及一个关于它的 更简洁的回答

除了 Thomas 的答案之外,静态变量存储在非堆区域中,这个区域称为方法区域。

类变量(静态变量)存储为与该类关联的 Class object的一部分。这个 Class 对象只能由 JVM 创建,并存储在 permanent generation中。

也有人回答说,它是存储在非堆区域,这是所谓的 Method Area.即使这个答案是没有错误的。Permgen Area 是否是堆的一部分,这是一个有争议的话题。显然每个人的看法不同。在我看来,我们在 JVM 参数中提供堆空间和 permgen 空间的方式不同。因此,区别对待它们是一个很好的假设。

换个角度看

内存池是由 JVM 内存管理器在运行时创建的。内存池可以属于堆内存,也可以属于非堆内存。运行时常量池是类文件中常量 _ pool 表的每个类或每个接口的运行时表示形式。每个运行时常量池都是从 Java 虚拟机的方法区域分配的,静态变量存储在这个方法区域中。 此外,这个非堆只是 perm gen 区域。实际上方法区域是 perm gen (参考文献)的一部分

enter image description here

由于静态变量是类级变量,它们将存储堆内存的“ 永久的一代”。 有关 JVM 的更多细节,请查看 这个。希望这会有所帮助

Java8之前:

静态变量存储在 permgen 空间(也称为方法区域)中。

PermGen 空间也被称为方法区域

PermGen 空间用来存储3样东西

  1. 类级别数据(元数据)
  2. 实习生字符串
  3. 静态变量

由 Java 8开始

静态变量存储在堆本身中。从 Java8开始,PermGen 空间被移除,并引入了名为 MetaSpace 的新空间,它不再是 Heap 的一部分,不像以前的 PermGen 空间。Meta-Space 存在于本机内存(由操作系统提供给特定应用程序供其使用的内存)中,现在它只存储类元数据。

内部的字符串和静态变量被移动到堆中。

有关官方信息,请参阅: JEP 122: 移除永久的发电空间

当我们创建一个静态变量或方法时,它被存储在堆的特殊区域: PermGen (永久生成) ,在那里它与应用到类的所有数据(非实例数据)一起放置。从 Java8开始,PermGen 变成了 Metaspace。不同之处在于 Metaspace 是自动增长的空间,而 PermGen 有一个固定的 Max 大小,并且这个空间在所有实例之间共享。另外,元空间是本机内存的一部分,而不是 JVM 内存的一部分。

您可以查看 这个了解更多细节。

在现实世界或项目中,我们提前有需求,需要在类中创建变量和方法,根据需求我们需要决定是否需要创建

  1. 本地(在块或方法构造函数中创建 n 个访问)
  2. 静电干扰,
  3. 实例变量(每个对象都有自己的副本) ,

= > 2.静态关键字将与变量一起使用,变量对所有对象的特定类都是相同的,

例如,在 selenium 中: 我们将 webDriver 标记为 static = > ,这样我们就不需要为每个测试用例一次又一次地创建 webDriver

Static Webdriver driver

(但并行执行会引起问题,但这是另一种情况) ;

真实世界的情景 = > 如果印度是阶级,那么国旗,金钱对每个印度人来说都是一样的,所以我们可以把它们看作是静态的。

另一个例子: 实用程序方法,我们总是声明为静态 b‘ ,因为它将在不同的测试用例中使用。 存储在 CMA (PreGen 空间) = PreGen (固定内存)中的静态数据在 java 8之后转移到了 Metaspace,因为现在它是动态增长的

从 Java8开始,PermGen 空间已经过时了。静态方法、基元和引用变量存储在 Java 元空间中。实际的对象驻留在 JAVA 堆中。因为静态方法从来没有脱离引用,所以它们从来没有从 MetaSpace 和 HEAP 收集到垃圾。