Java Virtual Machine - Menyetel GC
Pada bab terakhir, kita belajar tentang berbagai Gcs Generasi. Dalam bab ini, kita akan membahas tentang cara menyetel GC.
Ukuran Heap
Ukuran heap merupakan faktor penting dalam performa aplikasi Java kami. Jika terlalu kecil, maka akan sering diisi dan akibatnya harus sering dikumpulkan oleh GC. Di sisi lain, jika kita hanya menambah ukuran heap, meskipun perlu dikumpulkan lebih jarang, panjang jeda akan bertambah.
Lebih lanjut, meningkatkan ukuran heap memiliki hukuman berat pada OS yang mendasarinya. Menggunakan paging, OS membuat program aplikasi kita melihat lebih banyak memori daripada yang sebenarnya tersedia. OS mengatur ini dengan menggunakan beberapa ruang swap pada disk, menyalin bagian program yang tidak aktif ke dalamnya. Ketika bagian-bagian itu diperlukan, OS menyalinnya kembali dari disk ke memori.
Misalkan sebuah mesin memiliki memori 8G, dan JVM melihat memori virtual 16G, JVM tidak akan tahu bahwa sebenarnya hanya ada 8G yang tersedia pada sistem. Itu hanya akan meminta 16G dari OS, dan setelah mendapat memori itu, itu akan terus menggunakannya. OS harus menukar banyak data masuk dan keluar, dan ini adalah hukuman kinerja yang sangat besar pada sistem.
Dan kemudian muncul jeda yang akan terjadi selama GC penuh dari memori virtual tersebut. Karena GC akan bertindak di seluruh heap untuk pengumpulan dan pemadatan, GC harus menunggu lama hingga memori virtual ditukar keluar dari disk. Dalam kasus kolektor serentak, utas latar belakang harus menunggu lama untuk data disalin dari ruang swap ke memori.
Jadi di sini muncul pertanyaan tentang bagaimana kita harus memutuskan ukuran heap yang optimal. Aturan pertama adalah jangan pernah meminta OS lebih banyak memori daripada yang sebenarnya ada. Ini benar-benar akan mencegah masalah untuk sering bertukar. Jika mesin memiliki beberapa JVM yang terpasang dan berjalan, maka total permintaan memori oleh semuanya digabungkan kurang dari RAM aktual yang ada dalam sistem.
Anda dapat mengontrol ukuran permintaan memori oleh JVM menggunakan dua tanda -
-XmsN - Mengontrol memori awal yang diminta.
-XmxN - Mengontrol memori maksimum yang dapat diminta.
Nilai default dari kedua flag ini bergantung pada OS yang mendasarinya. Misalnya, untuk 64b JVM yang berjalan di MacOS, -XmsN = 64M dan -XmxN = minimal 1G atau 1/4 dari total memori fisik.
Perhatikan bahwa JVM dapat menyesuaikan di antara dua nilai secara otomatis. Misalnya, jika diketahui bahwa terlalu banyak GC yang terjadi, ukuran memori akan terus meningkat selama di bawah -XmxN dan sasaran performa yang diinginkan terpenuhi.
Jika Anda tahu persis berapa banyak memori yang dibutuhkan aplikasi Anda, maka Anda dapat mengatur -XmsN = -XmxN. Dalam hal ini, JVM tidak perlu mencari nilai "optimal" dari heap, dan karenanya, proses GC menjadi sedikit lebih efisien.
Ukuran Generasi
Anda dapat memutuskan berapa banyak heap yang ingin Anda alokasikan ke YG, dan berapa banyak yang ingin Anda alokasikan ke OG. Kedua nilai ini memengaruhi kinerja aplikasi kami dengan cara berikut.
Jika ukuran YG sangat besar, maka akan lebih jarang dikumpulkan. Ini akan menghasilkan lebih sedikit jumlah objek yang dipromosikan ke OG. Di sisi lain, jika Anda memperbesar ukuran OG terlalu banyak, maka pengumpulan dan pemadatan akan memakan banyak waktu dan ini akan menyebabkan jeda STW yang lama. Jadi, pengguna harus menemukan keseimbangan antara dua nilai ini.
Di bawah ini adalah flag yang dapat Anda gunakan untuk menyetel nilai-nilai ini -
-XX:NewRatio=N: Rasio YG terhadap OG (nilai default = 2)
-XX:NewSize=N: Ukuran awal YG
-XX:MaxNewSize=N: Ukuran maksimal YG
-XmnN: Setel NewSize dan MaxNewSize ke nilai yang sama menggunakan tanda ini
Ukuran awal YG ditentukan oleh nilai Rasio Baru dengan rumus yang diberikan -
(total heap size) / (newRatio + 1)
Karena nilai awal newRatio adalah 2, rumus di atas memberikan nilai awal YG menjadi 1/3 dari total ukuran heap. Anda selalu dapat mengganti nilai ini dengan secara eksplisit menentukan ukuran YG menggunakan flag NewSize. Bendera ini tidak memiliki nilai default, dan jika tidak disetel secara eksplisit, ukuran YG akan terus dihitung menggunakan rumus di atas.
Permagen dan Metaspace
Permagen dan metaspace adalah area heap tempat JVM menyimpan metadata kelas. Ruang ini disebut 'permagen' di Java 7, dan di Jawa 8 disebut 'metaspace'. Informasi ini digunakan oleh kompilator dan runtime.
Anda dapat mengontrol ukuran permagen menggunakan tanda berikut: -XX: PermSize=N dan -XX:MaxPermSize=N. Ukuran Metaspace dapat dikontrol menggunakan:-XX:Metaspace- Size=N dan -XX:MaxMetaspaceSize=N.
Ada beberapa perbedaan bagaimana permagen dan metaspace dikelola saat nilai bendera tidak disetel. Secara default, keduanya memiliki ukuran awal default. Namun sementara metaspace dapat menempati heap sebanyak yang dibutuhkan, permagen tidak dapat menempati lebih dari nilai awal default. Misalnya, 64b JVM memiliki ruang heap 82M sebagai ukuran permagen maksimum.
Perhatikan bahwa karena metaspace dapat menempati jumlah memori yang tidak terbatas kecuali ditentukan untuk tidak, dapat terjadi kesalahan kehabisan memori. GC penuh terjadi setiap kali wilayah ini diubah ukurannya. Karenanya, selama startup, jika ada banyak class yang dimuat, metaspace dapat terus mengubah ukuran sehingga menghasilkan GC penuh setiap saat. Oleh karena itu, dibutuhkan banyak waktu untuk aplikasi besar untuk memulai jika ukuran metaspace awal terlalu rendah. Merupakan ide bagus untuk menambah ukuran awal karena mengurangi waktu startup.
Meskipun permagen dan metaspace menyimpan metadata kelas, itu tidak permanen, dan ruang tersebut diambil kembali oleh GC, seperti dalam kasus objek. Ini biasanya terjadi pada aplikasi server. Setiap kali Anda membuat penerapan baru ke server, metadata lama harus dibersihkan karena pemuat kelas baru sekarang membutuhkan ruang. Ruang ini dibebaskan oleh GC.