ওহে! আজকের আলোচনায়, আমরা জাভাতে "ফ্যান্টম রেফারেন্স" (ফ্যান্টম রেফারেন্স) সম্পর্কে বিস্তারিত কথা বলব। এই ধরনের রেফারেন্স কি? কেন তাদের "ফ্যান্টম রেফারেন্স" বলা হয়? তারা কিভাবে ব্যবহার করা হয়? আপনি যেমন মনে রাখবেন, জাভাতে 4 ধরণের রেফারেন্স রয়েছে:
-
StrongReference (সাধারণ রেফারেন্স যা আমরা একটি বস্তু তৈরি করার সময় তৈরি করি):
Cat cat = new Cat()
এই উদাহরণে, বিড়াল একটি শক্তিশালী রেফারেন্স।
-
SoftReference (সফট রেফারেন্স)। আমরা যেমন রেফারেন্স সম্পর্কে একটি পাঠ ছিল.
-
দুর্বল রেফারেন্স (দুর্বল রেফারেন্স)। এখানে তাদের সম্পর্কে একটি পাঠ ছিল ।
-
ফ্যান্টম রেফারেন্স (ফ্যান্টম রেফারেন্স)।
-
get() — রেফারেন্সকৃত অবজেক্ট রিটার্ন করে;
- clear() — বস্তুর রেফারেন্স সরিয়ে দেয়।
finalize()
পদ্ধতি অবজেক্টের জন্য ওভাররাইড করা হয়েছে, এটি বলা হয়। অথবা হয়তো এটি বলা হয় না - এটি শুধুমাত্র আপনি ভাগ্যবান কিনা তা নির্ভর করে। আপনি সম্ভবত মনে রাখবেন যে finalize()
চঞ্চল :) আবর্জনা সংগ্রহকারীর দ্বিতীয় পাসে, বস্তুটি মুছে ফেলা হয় এবং মেমরি মুক্ত হয়। আবর্জনা সংগ্রহকারীর অপ্রত্যাশিত আচরণ আমাদের জন্য অনেক সমস্যার সৃষ্টি করে। আমরা ঠিক জানি না কখন আবর্জনা সংগ্রহকারী কাজ শুরু করবে। finalize()
পদ্ধতি বলা হবে কিনা আমরা জানি না । প্লাস, একটি বস্তুর একটি শক্তিশালী রেফারেন্স তৈরি করা যেতে পারে যখন এটির finalize()
পদ্ধতিটি কার্যকর করা হচ্ছে, এই ক্ষেত্রে বস্তুটি মুছে ফেলা হবে না। উপলব্ধ মেমরির উপর ভারী চাহিদা তৈরি করে এমন প্রোগ্রামগুলির জন্য, এটি সহজেই একটি OutOfMemoryError
. এই সব আমাদের ধাক্কা দেয় ফ্যান্টম রেফারেন্স ব্যবহার করতে. আসল বিষয়টি হ'ল এটি আবর্জনা সংগ্রহকারীর আচরণকে পরিবর্তন করে। যদি বস্তুর শুধুমাত্র ফ্যান্টম রেফারেন্স থাকে, তাহলে:
-
এর চূড়ান্তকরণ () পদ্ধতি বলা হয় (যদি এটি ওভাররাইড করা হয়)
-
ফাইনালাইজ() পদ্ধতি শেষ হওয়ার পরে যদি কিছুই পরিবর্তন না হয় এবং অবজেক্টটি এখনও মুছে ফেলা যায়, তাহলে অবজেক্টের ফ্যান্টম রেফারেন্স একটি বিশেষ সারিতে রাখা হয়: 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" অক্ষর যোগ করে) যাতে আরও মেমরি নেওয়া যায়। উপরন্তু, আমরা ফাইনালাইজ() পদ্ধতিটি ওভাররাইড করি যাতে এটি চালানো হয়। এর পরে, আমাদের একটি ক্লাস দরকার যা PhantomReference থেকে উত্তরাধিকারসূত্রে প্রাপ্ত হবে । কেন আমরা যেমন একটি ক্লাস প্রয়োজন? এটা সব সোজা. ফ্যান্টম রেফারেন্স সত্যিই সাফ করা হয়েছে কিনা তা যাচাই করার জন্য এটি আমাদের clear() পদ্ধতিতে অতিরিক্ত যুক্তি যোগ করার অনুমতি দেবে (যার অর্থ বস্তুটি মুছে ফেলা হয়েছে)।
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!
আমরা এখানে কি দেখতে পাচ্ছি? আমাদের পরিকল্পনা মতোই সবকিছু ঘটেছে! আমাদের অবজেক্টের ফাইনালাইজ() পদ্ধতিটি ওভাররাইড করা হয়েছে এবং আবর্জনা সংগ্রাহক চলাকালীন এটিকে কল করা হয়েছিল। এর পরে, ফ্যান্টম রেফারেন্সটি রেফারেন্স কিউতে রাখা হয়েছিল । সেখানে থাকাকালীন, এর ক্লিয়ার() পদ্ধতিকে কল করা হয়েছিল (যার মধ্যে আমরা কনসোলে আউটপুট দেওয়ার জন্য ক্লিনআপ() বলেছি)। অবশেষে, বস্তুটি মেমরি থেকে মুছে ফেলা হয়েছে। এখন আপনি দেখতে পাচ্ছেন যে এটি কীভাবে কাজ করে :) অবশ্যই, আপনাকে ফ্যান্টম রেফারেন্স সম্পর্কে সমস্ত তত্ত্ব মুখস্ত করতে হবে না। তবে অন্তত মূল বিষয়গুলো মনে রাখলে ভালো হবে। প্রথমত, এগুলি হল সবথেকে দুর্বল রেফারেন্স। এগুলি তখনই কার্যকর হয় যখন বস্তুর অন্য কোন উল্লেখ অবশিষ্ট থাকে না। আমরা উপরে যে রেফারেন্সগুলি দিয়েছি তা সবথেকে শক্তিশালী থেকে দুর্বলতম পর্যন্ত নিচের ক্রমে সাজানো হয়েছে: StrongReference -> SoftReference -> WeakReference -> PhantomReference একটি ফ্যান্টম রেফারেন্স তখনই যুদ্ধে প্রবেশ করে যখন আমাদের বস্তুর কোন শক্তিশালী, নরম বা দুর্বল রেফারেন্স না থাকে : ) দ্বিতীয়ত, 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