为什么叫做栈内存?难道此内存机制使用了栈的原理?
这是百科的解释,java可以换成任何编程语言 :C/PHP/Python
就好像我们在页面最顶部设置了一个变量
<code> $a = 1;(入) $b = 2;(入) print_r($a); print_r($b); </code>
难道$a = 1是到最后才输出的吗?
怎么回事呢?是我理解有误吗?
问题:为什么叫做栈内存?难道此内存机制使用了栈的原理?
回复内容:
这是百科的解释,java可以换成任何编程语言 :C/PHP/Python
就好像我们在页面最顶部设置了一个变量
<code> $a = 1;(入) $b = 2;(入) print_r($a); print_r($b); </code>
难道$a = 1是到最后才输出的吗?
怎么回事呢?是我理解有误吗?
问题:为什么叫做栈内存?难道此内存机制使用了栈的原理?
栈内存一般存储的是函数的调用信息和函数中申明的变量,因为函数的调用是递归的,外层函数一定比内层被调用的函数先加载和执行,而一定等到内层被调用函数结束后才能结束,这个先进后出的机制就是为什么叫栈内存的原因。
PS:在编译时编译器会先收集此函数中所有定义的变量,将他们放在函数最前面申请内存,所以他们进出栈的顺序不是你在编写程序时定义的顺序,而是在函数执行前进栈,函数执行完成后出栈。
举个实际的例子吧:
假设某个调用过程是这样的
<code>void a() { int p = 1; int q = 2; } void b() { int x = 3; int y = 4; a(); int z = 5; } </code>
那么我们在调用b();
的过程中栈内存其实经历了一下变化:
([a]
、[b]
代表a()
和b()
的基本信息,如程序指针等)
进入b函数时
栈底 (函数信息空间进栈)<br><code>[b]
为b函数内参数申请空间
栈底 (参数空间进栈)<br><code>[b]
其他操作无关进出栈,略去
进入a函数时 (函数信息空间进栈)
栈底 <br><code>[b]
为a函数内参数申请空间
栈底 (参数空间进栈)<br><code>[b]
完成a函数时 (参数空间出栈)
栈底 <br><code>[b]
退出a函数时 (函数信息空间出栈)
栈底 <br><code>[b]
完成b函数时 (参数空间出栈)
栈底 <br><code>[b]
退出b函数时 (函数信息空间出栈)
栈底 <br><code>
翻译害死人,这里真正的翻译叫“栈帧”,Stack Frame!
在JVM设计中一个方法是一个大的Stack Frame,当然方法内部还可以有多个小的Stack Frame,这里不展开讨论。
对于一个方法内部的临时变量而言,是分配在Stack Frame的局部变量表中的(可以理解为一个数组)Local Variable Table
比如这样一段代码,就会在将p和q两个局部变量丢到局部变量表中。当一个方法写完之后,局部变量表的大小也就随之确定了下来
<code>void a() { int p = 1; int q = 2; }</code>
接下来说到“栈”,为啥叫Stack Frame,其实这里表达的是一个Frame,Stack只是这个Frame的修饰定语而已。因为JVM是基于栈来完成指令运算操作的。(这里你可以去Google下两种不同的VM实现方式:基于寄存器和基于栈),JVM之所以选择基于栈来完成指令运算的设计结构,主要是考虑到了平台迁移的因素,因为不同的CPU架构下,寄存器的个数是不确定的,当然你也可以虚拟出寄存器来玩,但整体的实现成本就复杂上来了。
所以,大学的数据结构一定要学好。堆是堆、栈是栈,没有单独的堆栈这种数据结构描述。栈帧是帧,不是栈!
也许你需要去看一下CSAPP或者操作系统或者编译原理(实际上CSAPP就够XD)了……
这里的栈与数据结构的栈只是凑巧重名...

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック









JWTは、JSONに基づくオープン標準であり、主にアイデンティティ認証と情報交換のために、当事者間で情報を安全に送信するために使用されます。 1。JWTは、ヘッダー、ペイロード、署名の3つの部分で構成されています。 2。JWTの実用的な原則には、JWTの生成、JWTの検証、ペイロードの解析という3つのステップが含まれます。 3. PHPでの認証にJWTを使用する場合、JWTを生成および検証でき、ユーザーの役割と許可情報を高度な使用に含めることができます。 4.一般的なエラーには、署名検証障害、トークンの有効期限、およびペイロードが大きくなります。デバッグスキルには、デバッグツールの使用とロギングが含まれます。 5.パフォーマンスの最適化とベストプラクティスには、適切な署名アルゴリズムの使用、有効期間を合理的に設定することが含まれます。

このチュートリアルでは、PHPを使用してXMLドキュメントを効率的に処理する方法を示しています。 XML(拡張可能なマークアップ言語)は、人間の読みやすさとマシン解析の両方に合わせて設計された多用途のテキストベースのマークアップ言語です。一般的にデータストレージに使用されます

文字列は、文字、数字、シンボルを含む一連の文字です。このチュートリアルでは、さまざまな方法を使用してPHPの特定の文字列内の母音の数を計算する方法を学びます。英語の母音は、a、e、i、o、u、そしてそれらは大文字または小文字である可能性があります。 母音とは何ですか? 母音は、特定の発音を表すアルファベットのある文字です。大文字と小文字など、英語には5つの母音があります。 a、e、i、o、u 例1 入力:string = "tutorialspoint" 出力:6 説明する 文字列「TutorialSpoint」の母音は、u、o、i、a、o、iです。合計で6元があります

静的結合(静的::) PHPで後期静的結合(LSB)を実装し、クラスを定義するのではなく、静的コンテキストで呼び出しクラスを参照できるようにします。 1)解析プロセスは実行時に実行されます。2)継承関係のコールクラスを検索します。3)パフォーマンスオーバーヘッドをもたらす可能性があります。

Java 8は、Stream APIを導入し、データ収集を処理する強力で表現力のある方法を提供します。ただし、ストリームを使用する際の一般的な質問は次のとおりです。 従来のループにより、早期の中断やリターンが可能になりますが、StreamのForeachメソッドはこの方法を直接サポートしていません。この記事では、理由を説明し、ストリーム処理システムに早期終了を実装するための代替方法を調査します。 さらに読み取り:JavaストリームAPIの改善 ストリームを理解してください Foreachメソッドは、ストリーム内の各要素で1つの操作を実行する端末操作です。その設計意図はです

PHPの魔法の方法は何ですか? PHPの魔法の方法には次のものが含まれます。1。\ _ \ _コンストラクト、オブジェクトの初期化に使用されます。 2。\ _ \ _リソースのクリーンアップに使用される破壊。 3。\ _ \ _呼び出し、存在しないメソッド呼び出しを処理します。 4。\ _ \ _ get、dynamic属性アクセスを実装します。 5。\ _ \ _セット、動的属性設定を実装します。これらの方法は、特定の状況で自動的に呼び出され、コードの柔軟性と効率を向上させます。

PHP8では、一致式は、式の値に基づいて異なる結果を返す新しい制御構造です。 1)Switchステートメントに似ていますが、実行ステートメントブロックの代わりに値を返します。 2)一致式の式は厳密に比較され、セキュリティが向上します。 3)スイッチステートメントの脱落の可能性を回避し、コードのシンプルさと読みやすさを向上させます。

カプセルは3次元の幾何学的図形で、両端にシリンダーと半球で構成されています。カプセルの体積は、シリンダーの体積と両端に半球の体積を追加することで計算できます。このチュートリアルでは、さまざまな方法を使用して、Javaの特定のカプセルの体積を計算する方法について説明します。 カプセルボリュームフォーミュラ カプセルボリュームの式は次のとおりです。 カプセル体積=円筒形の体積2つの半球体積 で、 R:半球の半径。 H:シリンダーの高さ(半球を除く)。 例1 入力 RADIUS = 5ユニット 高さ= 10単位 出力 ボリューム= 1570.8立方ユニット 説明する 式を使用してボリュームを計算します。 ボリューム=π×R2×H(4
