हाय! आजच्या चर्चेत आपण जावा मधील "फँटम रेफरन्सेस" (फँटम रेफरन्स) बद्दल सविस्तर बोलू. हे कोणत्या प्रकारचे संदर्भ आहेत? त्यांना "फँटम रेफरन्स" का म्हणतात? ते कसे वापरले जातात? तुम्हाला आठवत असेल, Java मध्ये 4 प्रकारचे संदर्भ आहेत:
-
StrongReference (ऑब्जेक्ट तयार करताना आपण तयार करतो ते सामान्य संदर्भ):
Cat cat = new Cat()
या उदाहरणात, मांजर एक मजबूत संदर्भ आहे.
-
SoftReference (सॉफ्ट रेफरन्स). आमच्याकडे अशा संदर्भांचा धडा होता.
-
WeakReference (कमकुवत संदर्भ). येथे त्यांच्याबद्दल एक धडा देखील होता .
-
PhantomReference (फॅंटम संदर्भ).
-
get() — संदर्भित ऑब्जेक्ट परत करते;
- clear() — ऑब्जेक्टचा संदर्भ काढून टाकते.
finalize()
ऑब्जेक्टसाठी पद्धत ओव्हरराइड केली गेली आहे, त्याला म्हणतात. किंवा कदाचित ते म्हटले जात नाही - हे सर्व केवळ आपण भाग्यवान आहात की नाही यावर अवलंबून आहे. तुम्हाला कदाचित आठवत असेल की ते finalize()
चंचल आहे :) कचरा गोळा करणाऱ्याच्या दुसऱ्या पासमध्ये, ऑब्जेक्ट हटवला जातो आणि मेमरी मुक्त केली जाते. कचरा वेचणार्यांच्या अप्रत्याशित वागणुकीमुळे आमच्यासाठी अनेक समस्या निर्माण होतात. कचरा उचलण्याचे काम नेमके केव्हा सुरू होईल, हे कळत नाही. finalize()
पद्धत कॉल केली जाईल की नाही हे आम्हाला माहित नाही . शिवाय, ऑब्जेक्टची finalize()
पद्धत अंमलात आणली जात असताना त्याचा एक मजबूत संदर्भ तयार केला जाऊ शकतो, अशा परिस्थितीत ऑब्जेक्ट हटविला जाणार नाही. उपलब्ध मेमरीवर जास्त मागणी करणार्या प्रोग्रामसाठी, हे सहजपणे होऊ शकते OutOfMemoryError
. हे सर्व आपल्याला फॅन्टम संदर्भ वापरण्यास प्रवृत्त करते. वस्तुस्थिती अशी आहे की यामुळे कचरा वेचणाऱ्याच्या वागण्यात बदल होतो. जर ऑब्जेक्टमध्ये फक्त फॅन्टम संदर्भ असतील तर:
-
त्याची finalize() पद्धत म्हणतात (जर ती अधिलिखित केली असेल)
-
Finalize() पद्धत पूर्ण झाल्यावर काहीही बदलले नाही आणि ऑब्जेक्ट अजूनही हटवला जाऊ शकतो, तर ऑब्जेक्टचा फॅन्टम संदर्भ एका विशेष रांगेत ठेवला जातो: ReferenceQueue .
public class TestClass {
private StringBuffer data;
public TestClass() {
this.data = new StringBuffer();
for (long i = 0; i < 50000000; i++) {
this.data.append('x');
}
}
@Override
protected void finalize() {
System.out.println("The finalize method has been called on the TestClass object");
}
}
जेव्हा आम्ही वस्तू तयार करतो, तेव्हा अधिक मेमरी घेण्यासाठी आम्ही त्यांना हेतुपुरस्सर एक मोठा "भार" (प्रत्येक ऑब्जेक्टमध्ये 50 दशलक्ष "x" वर्ण जोडून) देऊ. या व्यतिरिक्त, ती रन झाली आहे हे पाहण्यासाठी आम्ही finalize() पद्धत ओव्हरराइड करतो. पुढे, आम्हाला PhantomReference कडून वारसा मिळेल अशा वर्गाची आवश्यकता आहे . आम्हाला अशा वर्गाची गरज का आहे? हे सर्व सरळ आहे. हे आम्हाला स्पष्ट () पद्धतीमध्ये अतिरिक्त तर्क जोडण्यास अनुमती देईल जेणेकरून फॅंटम संदर्भ खरोखर साफ झाला आहे (ज्याचा अर्थ ऑब्जेक्ट हटविला गेला आहे).
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
public class MyPhantomReference<TestClass> extends PhantomReference<TestClass> {
public MyPhantomReference(TestClass obj, ReferenceQueue<TestClass> queue) {
super(obj, queue);
Thread thread = new QueueReadingThread<TestClass>(queue);
thread.start();
}
public void cleanup() {
System.out.println("Cleaning up a phantom reference! Removing an object from memory!");
clear();
}
}
पुढे, आम्हाला एका वेगळ्या धाग्याची आवश्यकता आहे जो कचरा गोळा करणार्याचे काम करण्यासाठी प्रतीक्षा करेल आणि आमच्या संदर्भपंक्तीत फॅंटम लिंक्स दिसतील . रांगेत असा संदर्भ संपताच, त्यावर क्लीनअप() पद्धत कॉल केली जाते:
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
public class QueueReadingThread<TestClass> extends Thread {
private ReferenceQueue<TestClass> referenceQueue;
public QueueReadingThread(ReferenceQueue<TestClass> referenceQueue) {
this.referenceQueue = referenceQueue;
}
@Override
public void run() {
System.out.println("The thread monitoring the queue has started!");
Reference ref = null;
// Wait until the references appear in the queue
while ((ref = referenceQueue.poll()) == null) {
try {
Thread.sleep(50);
}
catch (InterruptedException e) {
throw new RuntimeException("Thread " + getName() + " was interrupted!");
}
}
// As soon as a phantom reference appears in the queue, clean it up
((MyPhantomReference) ref).cleanup();
}
}
आणि शेवटी, आपल्याला main() पद्धतीची आवश्यकता आहे, जी आपण त्यास वेगळ्या मुख्य वर्गात ठेवू . त्या पद्धतीमध्ये, आम्ही एक TestClass ऑब्जेक्ट, त्याचा एक फॅंटम संदर्भ आणि फॅंटम संदर्भांसाठी एक रांग तयार करू. त्यानंतर, आम्ही कचरा गोळा करणाऱ्याला कॉल करू आणि काय होते ते पाहू :)
import java.lang.ref.*;
public class Main {
public static void main(String[] args) throws InterruptedException {
Thread.sleep(10000);
ReferenceQueue<TestClass> queue = new ReferenceQueue<>();
Reference ref = new MyPhantomReference<>(new TestClass(), queue);
System.out.println("ref = " + ref);
Thread.sleep(5000);
System.out.println("Collecting garbage!");
System.gc();
Thread.sleep(300);
System.out.println("ref = " + ref);
Thread.sleep(5000);
System.out.println("Collecting garbage!");
System.gc();
}
}
कन्सोल आउटपुट:
ref = MyPhantomReference@4554617c
The thread monitoring the queue has started!
Collecting garbage!
The finalize method has been called on the TestClass object
ref = MyPhantomReference@4554617c
Collecting garbage!
Cleaning up a phantom reference!
Removing an object from memory!
आम्ही येथे काय पाहतो? आम्ही ठरवल्याप्रमाणे सर्व काही घडले! आमच्या ऑब्जेक्टची finalize() पद्धत ओव्हरराइड केली आहे आणि कचरा गोळा करणारे चालू असताना कॉल केले गेले. पुढे, फॅंटम संदर्भ संदर्भ रांगेत ठेवला गेला . तेथे असताना, तिची स्पष्ट() पद्धत कॉल केली गेली (ज्यामध्ये आम्ही कन्सोलवर आउटपुट करण्यासाठी क्लीनअप() म्हटले). शेवटी, ऑब्जेक्ट मेमरीमधून हटविला गेला. आता हे नक्की कसे कार्य करते ते तुम्ही पहा :) अर्थात, तुम्हाला फॅंटम संदर्भांबद्दलचे सर्व सिद्धांत लक्षात ठेवण्याची गरज नाही. परंतु किमान मुख्य मुद्दे लक्षात ठेवल्यास चांगले होईल. प्रथम, हे सर्व सर्वात कमकुवत संदर्भ आहेत. जेव्हा वस्तूचे इतर कोणतेही संदर्भ शिल्लक नसतात तेव्हाच ते कार्यात येतात. आम्ही वर दिलेल्या संदर्भांची यादी सर्वात मजबूत ते सर्वात कमकुवत अशा उतरत्या क्रमाने क्रमवारी लावली आहे: मजबूत संदर्भ -> सॉफ्टरेफरन्स -> कमजोर संदर्भ -> फॅंटमरेफरन्स फँटम संदर्भ केवळ तेव्हाच लढाईत प्रवेश करतो जेव्हा आमच्या ऑब्जेक्टसाठी कोणतेही मजबूत, मऊ किंवा कमकुवत संदर्भ नसतात: ) दुसरे, get() पद्धत फॅंटम संदर्भासाठी नेहमी शून्य देते . येथे एक साधे उदाहरण आहे जेथे आम्ही तीन वेगवेगळ्या प्रकारच्या कारसाठी तीन भिन्न प्रकारचे संदर्भ तयार करतो:
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.lang.ref.WeakReference;
public class Main {
public static void main(String[] args) {
Sedan sedan = new Sedan();
HybridAuto hybrid = new HybridAuto();
F1Car f1car = new F1Car();
SoftReference<Sedan> softReference = new SoftReference<>(sedan);
System.out.println(softReference.get());
WeakReference<HybridAuto> weakReference = new WeakReference<>(hybrid);
System.out.println(weakReference.get());
ReferenceQueue<F1Car> referenceQueue = new ReferenceQueue<>();
PhantomReference<F1Car> phantomReference = new PhantomReference<>(f1car, referenceQueue);
System.out.println(phantomReference.get());
}
}
कन्सोल आउटपुट:
Sedan@4554617c
HybridAuto@74a14482
null
get () पद्धतीने मऊ आणि कमकुवत संदर्भांसाठी संपूर्णपणे सामान्य वस्तू परत केल्या, परंतु ते फॅंटम संदर्भासाठी शून्य परत केले. तिसरे, फॅंटम संदर्भ मुख्यतः मेमरीमधून वस्तू हटविण्याच्या क्लिष्ट प्रक्रियेमध्ये वापरले जातात. बस एवढेच! :) आज आपला धडा संपतो. परंतु आपण एकट्या सिद्धांतावर फार पुढे जाऊ शकत नाही, म्हणून कार्य सोडवण्याकडे परत जाण्याची वेळ आली आहे! :)
GO TO FULL VERSION