- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
Java 应用程序启动时所有线程都有一个堆。每个线程都有自己的堆栈。
当一个Java应用程序启动时,我们使用JVM选项-Xms
和-Xmx
来控制堆的大小和-Xss
控制堆栈大小。
我的理解是,正在创建的堆成为 JVM 的“托管”内存,并且所有正在创建的对象都放置在那里。
但是堆栈创建是如何工作的? Java在创建每个线程时是否为每个线程创建一个堆栈?如果是这样,堆栈在内存中的确切位置?它肯定不在“托管”堆中。
JVM 是从 native 内存创建堆栈还是为堆栈预先分配一部分托管内存区域?如果是这样,JVM如何知道线程将如何创建?
最佳答案
Java specification 有一些关于线程堆栈的事情。告诉我们。除其他事项外:
每个 Java 虚拟机线程都有一个私有(private)的 Java 虚拟机堆栈,与线程同时创建。
因为除了推送和弹出帧外,Java 虚拟机堆栈永远不会被直接操作,因此帧可能是堆分配的。 Java 虚拟机堆栈的内存不需要是连续的。
规范允许 Java 虚拟机堆栈具有固定大小或根据计算需要动态扩展和收缩。
现在,如果我们专注于诸如 HotSpot 之类的 JVM 实现,我们可以获得更多信息。以下是我从不同来源收集的一些事实:
-Xss
选项的用途。 (Source) In Java SE 6, the default on Sparc is 512k in the 32-bit VM, and 1024k in the 64-bit VM. ... You can reduce your stack size by running with the -Xss option. ... 64k is the least amount of stack space allowed per thread.
Note that the JVM uses more memory than just the heap. For example Java methods, thread stacks and native handles are allocated in memory separate from the heap, as well as JVM internal data structures.
在 HotSpot 中,Java 线程和本地操作系统线程之间存在直接映射关系。 (Source) .
但 HotSpot 中的 Java 线程堆栈是软件管理的,它不是 OS 原生线程堆栈。 (Source)
It uses a separate software stack to pass Java arguments, while the native C stack is used by the VM itself. A number of JVM internal variables, such as the program counter or the stack pointer for a Java thread, are stored in C variables, which are not guaranteed to be always kept in the hardware registers. Management of these software interpreter structures consumes a considerable share of total execution time.
JVM 还为本地方法和 JVM 运行时调用(例如类加载)使用相同的 Java 线程堆栈。 (Source) .
有趣的是,即使分配的对象有时也可能位于堆栈而不是堆上,以作为性能优化。 (Source)
JVMs can use a technique called escape analysis, by which they can tell that certain objects remain confined to a single thread for their entire lifetime, and that lifetime is bounded by the lifetime of a given stack frame. Such objects can be safely allocated on the stack instead of the heap.
因为一张图片值一千字,这里有一张来自 James Bloom
现在回答您的一些问题:
How does JVM knows how may threads will be created?
它没有。通过创建可变数量的线程可以很容易地通过矛盾证明。它确实对最大线程数和每个线程的堆栈大小做出了一些假设。这就是为什么如果分配太多线程可能会耗尽内存(不是堆内存!)。
Does Java create a stack for each thread when it is created?
如前所述,每个 Java 虚拟机线程都有一个私有(private)的 Java 虚拟机堆栈,与线程同时创建。 (Source) .
If so, where exactly the stack is on the memory? It is certainly not in the "managed" heap.
如上所述,Java specification从技术上讲,允许堆栈内存存储在堆上。但至少 JRockit JVM 使用了不同的内存部分。
Does JVM create stack from native memory or does it pre-allocate a section of managed memory area for stack?
堆栈由 JVM 管理,因为 Java 规范 prescribes它必须如何表现:Java 虚拟机堆栈存储帧(第 2.6 节)。 Java 虚拟机堆栈类似于传统语言的堆栈。一个异常(exception)是用于 native
方法的 Native Method 堆栈。在 the specification 中再次了解更多信息.
关于java - Java(JVM)如何为每个线程分配堆栈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36898701/
我有一个应用程序,它会抛出 GKSession 并在各种条件下(连接超时、 session 失败等)创建一个新的 GKSession。不过,我遇到了内存泄漏问题,并且有时会在重新连接几次循环后崩溃。
比如我在宿主代码中有一个浮点指针 float *p 是否可以确定他指向的内存类型(设备/主机)? 最佳答案 在 UVA system 中, 运行时 API 函数 cudaPointerGetAttri
我已将项目转换为 .Net 4.0 并且以下代码不起作用: typeof(RuntimeTypeHandle).GetMethod("Allocate", BindingFlags.Instance
当我声明 char ch = 'ab' 时,ch 只包含 'b',为什么它不存储 'a'? #include int main() { char ch = 'ab'; printf("%c"
我对 Disk Sector 和 Block 有疑问。扇区是一个单位,通常为 512 字节或 1k、2k、4k 等取决于硬件。文件系统 block 大小是一组扇区大小。 假设我正在存储一个 5KB 的
假设我有 8 个人和5000 个苹果。 我想将所有苹果分发给所有 8 个人,这样我就没有苹果了。 但每个人都应该得到不同数量 将它们全部分发出去的最佳方式是什么? 我是这样开始的: let peopl
我正在构建的网站顶部有一个搜索栏。与 Trello 或 Gmail 类似,我希望当用户按下“/”键时,他们的焦点就会转到该搜索框。 我的 JavaScript 看起来像这样: document.onk
我有一小段代码: if (PZ_APP.dom.isAnyDomElement($textInputs)){ $textInputs.on("focus", function(){
我观察到以下行为。 接受了两个属性变量。 @property (nonatomic, retain) NSString *stringOne; @property (nonatomic, assign
我正在解决这样的问题 - 实现一个计算由以下内容组成的表达式的函数以下操作数:“(”、“)”、“+”、“-”、“*”、“/”。中的每个数字表达式可能很大(与由字符串表示的一样大)1000 位)。 “/
我有一组主机和一组任务。 每个主机都有 cpu、mem 和任务容量,每个任务都有 cpu、mem 要求。 每个主机都属于一个延迟类别,并且可以与具有特定延迟值的其他主机通信。 每个任务可能需要以等于或
该程序的作用:从文件中读取一个包含 nrRows 行和 nrColomns 列的矩阵(二维数组)。矩阵的所有元素都是 [0,100) 之间的整数。程序必须重新排列矩阵内的所有元素,使每个元素等于其所在
世界!我有个问题。今天我尝试创建一个代码,它可以找到加泰罗尼亚语号码。但是在我的程序中可以是长数字。我找到了分子和分母。但我不能分割长数字!此外,只有标准库必须在此程序中使用。请帮帮我。这是我的代码
我确定我遗漏了一些明显的东西,但我想在 Objective C 中创建一个 NSInteger 指针的实例。 -(NSInteger*) getIntegerPointer{ NSInteger
这个问题在这里已经有了答案: Difference between self.ivar and ivar? (4 个答案) 关闭 9 年前。
我如何将 v[i] 分配给一系列整数(v 的类型是 vector )而无需最初填充 最佳答案 你的意思是将 std::vector 初始化为一系列整数? int i[] = {1, 2, 3, 4,
我想寻求分配方面的帮助....我把这个作业带到了学校......我必须编写程序来加载一个 G 矩阵和第二个 G 矩阵,并搜索第二个 G 矩阵以获取存在数第一个 G 矩阵的......但是,当我尝试运行
我必须管理资源。它基本上是一个唯一的编号,用于标识交换机中的第 2 层连接。可以有 16k 个这样的连接,因此每次用户希望配置连接时,他/她都需要分配一个唯一索引。同样,当用户希望删除连接时,资源(号
是否有任何通用的命名约定来区分已分配和未分配的字符串?我正在寻找的是希望类似于 us/s 来自 Making Wrong Code Look Wrong ,但我宁愿使用常见的东西也不愿自己动手。 最佳
我需要读取一个 .txt 文件并将文件中的每个单词分配到一个结构中,该结构从结构 vector 指向。我将在下面更好地解释。 感谢您的帮助。 我的程序只分配文件的第一个字... 我知道问题出在函数 i
我是一名优秀的程序员,十分优秀!