WeakHashMap のコードを調査して、より多くの知識を得るWeakReference
エントリが次のようになっていることがわかりました。
private static class Entry<K,V> extends WeakReference<Object> implements Map.Entry<K,V> {
V value;
final int hash;
Entry<K,V> next;
/**
* Creates new entry.
*/
Entry(Object key, V value,
ReferenceQueue<Object> queue,
int hash, Entry<K,V> next) {
super(key, queue);
this.value = value;
this.hash = hash;
this.next = next;
}
...
したがって、新しいエントリを作成するときは、 を呼び出しsuper(key, queue);
ます。WeakReference
コンストラクタです。オブジェクトがGCによって収集された後、私が理解している限り、新しい参照(上の参照であるべきだと思いますkey
)がキューに表示されます。
また、各操作で呼び出すメソッドに気付きました:
/**
* Expunges stale entries from the table.
*/
private void expungeStaleEntries() {
for (Object x; (x = queue.poll()) != null; ) {
synchronized (queue) {
@SuppressWarnings("unchecked")
Entry<K,V> e = (Entry<K,V>) x;
int i = indexFor(e.hash, table.length);
Entry<K,V> prev = table[i];
Entry<K,V> p = prev;
while (p != null) {
Entry<K,V> next = p.next;
if (p == e) {
if (prev == e)
table[i] = next;
else
prev.next = next;
// Must not null out e.next;
// stale entries may be in use by a HashIterator
e.value = null; // Help GC
size--;
break;
}
prev = p;
p = next;
}
}
}
}
(Entry<K,V>)
キューから取得したよう です。これを説明する方法がわかりません(最初の質問)。このコード:
public static void main(String[] args) throws InterruptedException {
StringBuilder AAA = new StringBuilder();
ReferenceQueue queue = new ReferenceQueue();
WeakReference weakRef = new WeakReference(AAA, queue);
AAA = null;
System.gc();
Reference removedReference = queue.remove();
System.out.println(removedReference.get());
}
オブジェクトはすでにGCによって収集されているため、常にnullを出力します
また、私にとっては、GC によって既に収集されたオブジェクトを参照できることも奇妙でした。実際には、参照がキューに表示されるはずですが、オブジェクトが既に収集されているため、意味のあるものを読み取ることができませんでした ( 2 番目の質問)。