Bộ nhớ Heap và Stack trong Java
09:33Bài
viết giải thích sự khác nhau của Heap và Stack trong Java, cũng như sự liên quan
giữa chúng trong quá trình chạy chương trình.
Thuật
ngữ sử dụng:
- Runtime:
thời gian chạy chương trình
- Java
Heap Memory:
Bộ nhớ Heap trong Java
- Java
Stack Memory:
Bộ nhớ Stack trong Java
- JVM:
Java Virtual Machine, máy ảo Java dùng để chạy chương trình Java của
bạn.
- Object:
là đối tượng được khởi tạo từ khoá new từ
một class.
1. Định
nghĩa
Java
Heap và Stack Memory là một phần của bộ nhớ được JVM sử dụng để chạy chương
trình Java của bạn. Khi
bạn chạy chương trình Java, JVM sẽ yêu cầu hệ điều hành cấp cho một không gian
bộ nhớ trong RAM để dùng cho việc chạy chương trình. JVM sẽ chia bộ nhớ được cấp
phát này thành 2 phần: Heap và Stack cho việc quản lý.
2. Java Heap
Memory
-
Là bộ nhớ được sử dụng ở runtime để lưu các Objects. Bắt cứ khi nào ở đâu trong
chương trình của bạn khi bạn tạo Object thì nó sẽ được lưu trong
Heap.
-
Các objects trong Heap đều được truy cập bởi tất cả các các nơi trong ứng dụng,
bởi các threads khác nhau.
-
Thời gian sống của object phụ thuộc vào Garbage Collection của java.
-
Garbage Collection sẽ chạy trên bộ nhớ Heap để xoá các Object không được sử dụng
nữa, nghĩa là object không được referece trong chương trình.
-
Dung lượng sử dụng của Heap sẽ tăng giảm phụ thuộc vào Objects sử
dụng.
-
Dung lượng Heap thường lớn hơn Stack.
3.
Java Stack Memory
-
Bộ nhớ để lưu các biến local trong hàm và lời gọi hàm ở runtime trong một Thread
java.
-
Các biến local bao gồm loại nguyên thuỷ (primitive) và loại tham chiếu tới đối
tượng trong heap (reference) khai báo trong hàm, hoặc đối số được truyền vào
hàm, thường có thời gian sống ngắn.
-
Bố nhớ stack thường nhỏ
-
Cơ chế hoạt động là LIFO (Last-In-First-Out), chạy sau chết trước.
-
Bất cứ khi nào gọi 1 hàm, một khối bộ nhớ mới sẽ được tạo trong Stack cho hàm đó
để lưu các biến local. Khi hàm thực hiện xong, khối bộ nhớ cho hàm sẽ bị xoá, và
giải phóng bộ nhớ trong stack.
3.
Ví dụ
-
Một chương trình java đơn giản.
Memory.java
package com.journaldev.test; public class Memory { public static void main(String[] args) { // Line 1 int i=1; // Line 2 Object obj = new Object(); // Line 3 Memory mem = new Memory(); // Line 4 mem.foo(obj); // Line 5 } // Line 9 private void foo(Object param) { // Line 6 String str = param.toString(); //// Line 7 System.out.println(str); } // Line 8 }
- Giải thích:
+ Khi chạy chương trình, một thread sẽ khởi tạo và sẽ gọi hàm main ở dòng 1. Một khối bộ nhớ được tạo trong stack cho hàm main().
+ một biến local được tạo ở dòng 2, loại primitive được lưu trong cùng khối bộ nhớ của hàm main()
+ Dòng 3, một đối tượng được tạo loại Object sẽ được lưu trong bộ nhớ heap và biến tham chiếu obj được lưu trong stack của hàm main()
+ Tương tự dòng 4, một đối tượng được tạo loại Memory lưu trong bộ nhớ heap và biến tham chiếu mem lưu trong stack.
+ Dòng 5 , hàm foo() thì được gọi, vì vậy nó sẽ tạo một khối mới bộ nhớ trong stack cho hàm foo()
+ Hàm foo có một đối số ở dòng 6, loại biến tham chiếu param tời đối tượng Object trong heap, nên đối số này sẽ được lưu trong bộ nhớ stack của foo()
+ Dòng 7, một biến local loại tham chiếu là str được tạo trong stack của foo(). Biến này tham chiếu tới đối tượng StringPool trong heap.
+ Dòng 8, hàm foo() sẽ kết thúc vì vậy bộ nhớ trong stack cho hàm foo() sẽ được giải phóng.
+ theo quy luật LIFO, foo() vào sau chết trước, và sau đó hàm main() cũng kết thúc, bộ nhớ trong stack cho hàm main() cũng được giải phóng.
Chương trình kết thúc.
4. Sự khác nhau stack và heap
- Heap được sử dụng bởi tất cả các phần, các luồng Thread của ứng dụng, còn stack chỉ được sử dụng cho một Thread duy nhất. Thread ngoài không thể truy cập vào được.
- Khi object được tạo nó lưu ở heap. stack chỉ lưu các biến local, đối số truyền vào hàm và lời gọi hàm, biến local có thể là loại primitive hoặc loại references (tham chiếu)
- Objects lưu trong heap được truy cập bởi tất cả các thread, còn trong stack thì không thể. Chỉ thread của stack đó mới truy cập được nó.
- Cơ chế quản lý của Stack là LIFO, còn Heap thì phức tạp hơn. Heap đuơc phân làm 2 loại Young-Generation, Old-Generation. Đọc thêm về Garbage Collection để hiểu rõ hơn.
- Thời gian sống của bộ nhớ Stack ngắn hơn so với heap.
- Sử dụng -Xms và -Xmx để định nghĩa dung lượng bắt đầu và dung lượng tối đa của bộ nhớ heap. Dùng -Xss để định nghĩa dung lượng bộ nhớ stack.
- Khi stack bị đầy bộ nhớ, chương trình phát sinh lỗi java.lang.StackOverFlowError, còn khi Heap bị đầy chương trình hiện lỗi java.lang.OutOfMemoryError: Java Heap Space
- Bộ nhớ Stack thường nhỏ hơn rất nhiều so với Heap, cũng như truy cập nhanh hơn bộ nhớ heap.
Hi vọng các bạn lập trình java sẽ nắm rõ hơn về 2 loại bộ nhớ trong Java, từ đó có thể hiểu rõ cơ chế, nắm được các lỗi phát sinh từ bộ nhớ để tuỳ chỉnh.
0 nhận xét