【问题标题】:Where are static methods and static variables stored in Java?Java中的静态方法和静态变量存储在哪里?
【发布时间】:2012-01-13 08:25:52
【问题描述】:

例如:

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 中的什么位置、堆内存或堆栈内存中?它们是如何存储的?

【问题讨论】:

标签: java jvm


【解决方案1】:

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

请注意,这主要适用于 Oracle 的 Hotspot 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 的版本之间也可能不同.以上基于 Java 5 和 6 的 HotSpot(它们基本相同),因为在回答时我会说大多数人都使用这些 JVM。由于自 Java 8 起内存模型发生了重大变化,上述陈述可能不适用于 Java 8 HotSpot - 而且我没有检查 Java 7 HotSpot 的变化,所以我猜测以上该版本仍然适用,但我不确定。

【讨论】:

  • 啊,你确定静态变量吗? AFAIK PermGen 只存储定义而不是实际值。
  • @Amir 我很确定变量本身存储在 permgen 空间中,任何引用的对象很可能会在堆上分配。这可能会添加一些信息:stackoverflow.com/questions/3800444/…
  • 是的,变量定义存储在 permgen 中。但是该值将在堆中。您的回答表明该值也存储在 PermGen 中。
  • @Matthew 你怎么理解我的回答? A 说 变量 存储在 permgen 部分(原语/引用)而不是它们引用的对象。这取决于您如何查看变量
  • @Nav 默认情况下,并非堆的所有部分都被垃圾收集,有时类和静态变量无法被收集,因为类加载器仍然对它们有引用。此外,您不应该依赖垃圾收集器来运行,因为这完全取决于 JVM(它决定何时运行以及收集什么,您只能提供“我希望您现在运行 gc”之类的提示 :)) .
【解决方案2】:

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

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

=>2。我们将使用静态关键字与变量一起使用,该变量对于所有对象的特定类都相同,例如在 selenium 中:我们将 webDriver 标记为 static=>,因此我们不需要为每个测试用例一次又一次地创建 webdriver= Static Webdriver driver(但并行执行会导致问题,但那是另一种情况);然后,现实世界场景=>如果印度是阶级,那么,旗帜,金钱将是每个印度人都一样,所以我们可能会认为是静态的。 Anatoher 示例:我们总是将实用方法声明为静态 b'cos,它将用于不同的测试用例。 静态存储在 CMA(PreGen 空间)=PreGen(固定内存)在 Java8 之后更改为 Metaspace,现在它动态增长

【讨论】:

    【解决方案3】:

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

    您可以查看this了解更多详情。

    【讨论】:

      【解决方案4】:

      Java 8 之前的版本:

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

      PermGen Space is also known as Method Area

      用于存储 3 个东西的 PermGen 空间

      1. 类级数据(元数据)
      2. 内部字符串
      3. 静态变量

      从 Java 8 开始

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

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

      官方信息请参考:JEP 122:Remove the Permanent Gen Space

      【讨论】:

      • 当你说静态变量的“堆本身”时>Java8,具体在哪里:OldGen?
      【解决方案5】:

      由于静态变量是类级别的变量,它们将存储“永久代”的堆内存。 请查看this 以获取有关 JVM 的更多详细信息。希望这会有所帮助

      【讨论】:

        【解决方案6】:

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

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

        另一种观看方式

        内存池由 JVM 内存管理器在运行时创建。内存池可能属于堆内存或非堆内存。运行时常量池是类文件中常量池表的按类或按接口运行时表示。每个运行时常量池都是从 Java 虚拟机的方法区分配的,静态变量存储在这个方法区中。 此外,这个非堆只不过是 perm gen 区域。实际上 Method area 是 perm gen 的一部分。(Reference

        【讨论】:

        • 方法区不是内存 PermGen 部分的子集吗?当我认为它们(PermGen 和方法(类)区域)是 JVM 较大堆区域的一部分时,为什么要将方法区域显示为非堆内存的一部分?
        • 阅读最后一行 - Also this non-heap is nothing but perm gen area.Actually Method area is part of perm gen.
        • @AniketThakur 您已将方法区显示为非堆内存的一部分,但根据 oracle 文档,这里 docs.oracle.com/javase/specs/jvms/se7/html/… 提到方法区在逻辑上是堆的一部分。
        【解决方案7】:

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

        【讨论】:

          【解决方案8】:

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

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

          长答案已经在堆栈溢出:

          有一个thorough description of memory and garbage collection in the JVM 和一个answer that talks more concisely 关于它。

          【讨论】:

          • 没问题!如果你觉得这些人有用,别忘了给他们点赞。
          【解决方案9】:

          静态变量存储在堆中

          【讨论】:

          • 静态变量存储在内存的PremGen空间中,它们的值存储在Heap中。
          【解决方案10】:

          它存储在类定义引用的堆中。如果您考虑一下,它与堆栈无关,因为没有范围。

          【讨论】:

          • 不正确。静态字段本身并不存储在堆中。
          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 2018-08-14
          • 2010-09-10
          • 2015-11-03
          相关资源
          最近更新 更多