जावा वर्चुअल मशीन - जीसी ट्यूनिंग
पिछले अध्याय में, हमने विभिन्न जनरेशन Gcs के बारे में सीखा। इस अध्याय में, हम चर्चा करेंगे कि जीसी को कैसे ट्यून किया जाए।
हीप साइज
ढेर का आकार हमारे जावा अनुप्रयोगों के प्रदर्शन का एक महत्वपूर्ण कारक है। यदि यह बहुत छोटा है, तो यह बार-बार भरा जाएगा और परिणामस्वरूप, जीसी द्वारा बार-बार एकत्र करना होगा। दूसरी ओर, अगर हम सिर्फ ढेर का आकार बढ़ाते हैं, हालांकि इसे कम बार इकट्ठा करने की आवश्यकता होती है, तो ठहराव की लंबाई बढ़ जाएगी।
इसके अलावा, ढेर के आकार को बढ़ाने से अंतर्निहित ओएस पर एक गंभीर जुर्माना होता है। पेजिंग का उपयोग करते हुए, ओएस हमारे एप्लिकेशन प्रोग्रामों को वास्तव में उपलब्ध होने की तुलना में बहुत अधिक मेमोरी देखता है। ओएस डिस्क पर कुछ स्वैप स्थान का उपयोग करके इसे प्रबंधित करता है, इसमें कार्यक्रमों के निष्क्रिय भागों की नकल करता है। जब उन हिस्सों की आवश्यकता होती है, तो ओएस उन्हें डिस्क से मेमोरी में वापस कॉपी करता है।
मान लें कि एक मशीन में 8G मेमोरी है, और JVM 16G वर्चुअल मेमोरी को देखता है, JVM को पता नहीं होगा कि वास्तव में सिस्टम पर केवल 8G उपलब्ध है। यह ओएस से केवल 16 जी का अनुरोध करेगा, और एक बार यह मेमोरी मिल जाने के बाद, यह इसका उपयोग करना जारी रखेगा। ओएस को अंदर और बाहर बहुत सारे डेटा को स्वैप करना होगा, और यह सिस्टम पर एक विशाल प्रदर्शन जुर्माना है।
और फिर पॉज़ेस आता है जो इस तरह की वर्चुअल मेमोरी के पूर्ण जीसी के दौरान होता है। चूंकि जीसी संग्रह और संघनन के लिए पूरे ढेर पर काम करेगा, इसलिए इसे वर्चुअल मेमोरी के डिस्क से बाहर स्वैप करने के लिए बहुत इंतजार करना होगा। समवर्ती कलेक्टर के मामले में, पृष्ठभूमि थ्रेड्स को स्वैप स्थान से मेमोरी में कॉपी किए जाने वाले डेटा के लिए बहुत इंतजार करना होगा।
तो यहाँ सवाल है कि हमें इष्टतम हीप आकार पर कैसे निर्णय लेना चाहिए। पहला नियम यह है कि वास्तव में मौजूद ओएस से अधिक मेमोरी का अनुरोध न करें। यह पूरी तरह से स्वैपिंग के लिए समस्या को पूरी तरह से रोक देगा। यदि मशीन में कई JVM स्थापित और चल रहे हैं, तो उन सभी द्वारा संयुक्त कुल मेमोरी अनुरोध सिस्टम में मौजूद वास्तविक रैम से कम है।
आप JVM द्वारा दो झंडों का उपयोग करके मेमोरी अनुरोध के आकार को नियंत्रित कर सकते हैं -
-XmsN - अनुरोध की गई प्रारंभिक मेमोरी को नियंत्रित करता है।
-XmxN - अनुरोध की जा सकने वाली अधिकतम मेमोरी को नियंत्रित करता है।
इन दोनों झंडों का डिफ़ॉल्ट मान अंतर्निहित OS पर निर्भर करता है। उदाहरण के लिए, Macb पर चलने वाली 64b JVM के लिए, -XmsN = 64M और -XmxN = कुल भौतिक मेमोरी का 1G या 1 / 4th न्यूनतम।
ध्यान दें कि JVM दो मानों के बीच स्वचालित रूप से समायोजित कर सकता है। उदाहरण के लिए, यदि यह नोटिस करता है कि बहुत अधिक जीसी हो रहा है, तो यह मेमोरी साइज को तब तक बढ़ाता रहेगा जब तक यह अंडरएक्सएमएक्सएन है और वांछित प्रदर्शन लक्ष्यों को पूरा किया जाता है।
यदि आपको पता है कि आपके एप्लिकेशन को कितनी मेमोरी की आवश्यकता है, तो आप -XmsN = -XmxN सेट कर सकते हैं। इस मामले में, जेवीएम को ढेर के "इष्टतम" मूल्य का पता लगाने की आवश्यकता नहीं है, और इसलिए, जीसी प्रक्रिया थोड़ी अधिक कुशल हो जाती है।
पीढ़ी आकार
आप यह तय कर सकते हैं कि आप YG को कितना आवंटित करना चाहते हैं और आप OG को कितना आवंटित करना चाहते हैं। ये दोनों मान निम्न प्रकार से हमारे अनुप्रयोगों के प्रदर्शन को प्रभावित करते हैं।
यदि YG का आकार बहुत बड़ा है, तो इसे कम बार एकत्र किया जाएगा। इससे OG को बढ़ावा देने वाली वस्तुओं की संख्या कम होगी। दूसरी ओर, यदि आप ओजी के आकार में बहुत अधिक वृद्धि करते हैं, तो इसे इकट्ठा करने और कॉम्पैक्ट करने में बहुत अधिक समय लगेगा और इससे लंबे समय तक एसटीडब्ल्यू रुक जाएगा। इस प्रकार, उपयोगकर्ता को इन दो मूल्यों के बीच एक संतुलन खोजना होगा।
नीचे वे झंडे हैं जिनका उपयोग आप इन मूल्यों को निर्धारित करने के लिए कर सकते हैं -
-XX:NewRatio=N: OG को YG का अनुपात (डिफ़ॉल्ट मान = 2)
-XX:NewSize=N: YG का प्रारंभिक आकार
-XX:MaxNewSize=N: YG का अधिकतम आकार
-XmnN: NewSize और MaxNewSize को इस ध्वज का उपयोग करके समान मान पर सेट करें
YG का प्रारंभिक आकार न्यूट्रियो के मान द्वारा दिए गए सूत्र द्वारा निर्धारित किया जाता है -
(total heap size) / (newRatio + 1)
चूंकि newRatio का प्रारंभिक मूल्य 2 है, इसलिए उपरोक्त सूत्र कुल ढेर आकार का 1/3 होने के लिए YG का प्रारंभिक मूल्य देता है। आप NewSize ध्वज का उपयोग करके YG के आकार को स्पष्ट रूप से निर्दिष्ट करके इस मान को हमेशा ओवरराइड कर सकते हैं। इस ध्वज का कोई डिफ़ॉल्ट मान नहीं है, और यदि यह स्पष्ट रूप से सेट नहीं किया गया है, तो YG का आकार उपरोक्त सूत्र का उपयोग करके गणना करता रहेगा।
पर्मगेन और मेटस्पैस
परमेजन और मेटासैप्स ऐसे ढेर क्षेत्र हैं जहां जेवीएम कक्षाओं की मेटाडेटा रखता है। अंतरिक्ष को जावा 7 में 'परमेजन' कहा जाता है, और जावा 8 में इसे 'मेटासैपेस' कहा जाता है। इस जानकारी का उपयोग कंपाइलर और रनटाइम द्वारा किया जाता है।
आप निम्न झंडों का उपयोग करके परिमेय के आकार को नियंत्रित कर सकते हैं: -XX: PermSize=N तथा -XX:MaxPermSize=N। मेटस्पैस के आकार का उपयोग करके नियंत्रित किया जा सकता है:-XX:Metaspace- Size=N तथा -XX:MaxMetaspaceSize=N।
कुछ मान हैं कि कैसे फ्लैग वैल्यू सेट नहीं होने पर परमेगन और मेटास्टेस को मैनेज किया जाता है। डिफ़ॉल्ट रूप से, दोनों का डिफ़ॉल्ट प्रारंभिक आकार होता है। लेकिन जब मेटासपेस जरूरत के रूप में ढेर के रूप में कब्जा कर सकता है, तो permagen डिफ़ॉल्ट प्रारंभिक मानों से अधिक नहीं रह सकता है। उदाहरण के लिए, 64b JVM में अधिकतम परमिट आकार के रूप में 82M ढेर स्थान है।
ध्यान दें कि चूंकि मेटास्टेस असीमित मात्रा में मेमोरी पर कब्जा कर सकता है जब तक कि निर्दिष्ट नहीं किया जाता है, मेमोरी त्रुटि से बाहर हो सकता है। जब भी इन क्षेत्रों का आकार बदला जा रहा है, तब एक पूर्ण जीसी होता है। इसलिए, स्टार्टअप के दौरान, यदि बहुत सारी कक्षाएं हैं जो लोड हो रही हैं, तो मेटासैप्स हर बार पूर्ण जीसी के परिणामस्वरूप आकार बदलते रह सकते हैं। इस प्रकार, बड़े अनुप्रयोगों के लिए स्टार्टअप के लिए बहुत समय लगता है यदि प्रारंभिक मेटास्टेस आकार बहुत कम है। शुरुआती आकार को बढ़ाने के लिए यह एक अच्छा विचार है क्योंकि यह स्टार्टअप के समय को कम करता है।
यद्यपि परमेजन और मेटास्टेस क्लास मेटाडेटा को धारण करते हैं, यह स्थायी नहीं है, और वस्तुओं के मामले में, जीसी द्वारा अंतरिक्ष को पुनः प्राप्त किया जाता है। यह आमतौर पर सर्वर एप्लिकेशन के मामले में होता है। जब भी आप सर्वर पर एक नई तैनाती करते हैं, तो पुराने मेटाडेटा को साफ करना होगा क्योंकि नए क्लास लोडर को अब जगह की आवश्यकता होगी। इस स्थान को जीसी द्वारा मुक्त किया गया है।