/* * Copyright 2000-2015 JetBrains s.r.o. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.intellij.util.containers; import gnu.trove.THashSet; import org.jetbrains.annotations.NotNull; import java.lang.ref.ReferenceQueue; import java.util.*; /** * Base class for concurrent key:int -> (weak/soft) value:V map * Null values are NOT allowed */ abstract class ConcurrentRefValueIntObjectHashMap<V> implements ConcurrentIntObjectMap<V> { private final ConcurrentIntObjectMap<IntReference<V>> myMap = ContainerUtil.createConcurrentIntObjectMap(); private final ReferenceQueue<V> myQueue = new ReferenceQueue<V>(); @NotNull protected abstract IntReference<V> createReference(int key, @NotNull V value, @NotNull ReferenceQueue<V> queue); interface IntReference<V> { int getKey(); V get(); } private void processQueue() { while (true) { @SuppressWarnings("unchecked") IntReference<V> ref = (IntReference)myQueue.poll(); if (ref == null) { return; } int key = ref.getKey(); myMap.remove(key, ref); } } @NotNull @Override public V cacheOrGet(int key, @NotNull V value) { processQueue(); IntReference<V> newRef = createReference(key, value, myQueue); while (true) { IntReference<V> ref = myMap.putIfAbsent(key, newRef); if (ref == null) return value; // there were no previous value V old = ref.get(); if (old != null) return old; // old value has been gced; need to overwrite boolean replaced = myMap.replace(key, ref, newRef); if (replaced) { return value; } } } @Override public boolean remove(int key, @NotNull V value) { processQueue(); return myMap.remove(key, createReference(key, value, myQueue)); } @Override public boolean replace(int key, @NotNull V oldValue, @NotNull V newValue) { processQueue(); return myMap.replace(key, createReference(key, oldValue,myQueue), createReference(key, newValue,myQueue)); } @Override public V put(int key, @NotNull V value) { processQueue(); IntReference<V> ref = myMap.put(key, createReference(key, value, myQueue)); return ref == null ? null : ref.get(); } @Override public V get(int key) { IntReference<V> ref = myMap.get(key); return ref == null ? null : ref.get(); } @Override public V remove(int key) { processQueue(); IntReference<V> ref = myMap.remove(key); return ref == null ? null : ref.get(); } @Override public boolean containsKey(int key) { return myMap.containsKey(key); } @Override public void clear() { myMap.clear(); processQueue(); } @NotNull @Override public int[] keys() { return myMap.keys(); } @NotNull @Override public Iterable<IntEntry<V>> entries() { final Iterator<IntEntry<IntReference<V>>> entryIterator = myMap.entries().iterator(); return new Iterable<ConcurrentIntObjectMap.IntEntry<V>>() { @Override public Iterator<ConcurrentIntObjectMap.IntEntry<V>> iterator() { return new Iterator<IntEntry<V>>() { private IntEntry<V> next = nextAliveEntry(); @Override public boolean hasNext() { return next != null; } @Override public IntEntry<V> next() { if (!hasNext()) throw new NoSuchElementException(); IntEntry<V> result = next; next = nextAliveEntry(); return result; } private IntEntry<V> nextAliveEntry() { while (entryIterator.hasNext()) { IntEntry<IntReference<V>> entry = entryIterator.next(); final V v = entry.getValue().get(); if (v == null) { continue; } final int key = entry.getKey(); return new IntEntry<V>() { @Override public int getKey() { return key; } @NotNull @Override public V getValue() { return v; } }; } return null; } @Override public void remove() { throw new UnsupportedOperationException(); } }; } }; } @Override public int size() { processQueue(); return myMap.size(); } @Override public boolean isEmpty() { processQueue(); return myMap.isEmpty(); } @NotNull @Override public Enumeration<V> elements() { final Enumeration<IntReference<V>> elementRefs = myMap.elements(); return new Enumeration<V>() { private V findNextRef() { while (elementRefs.hasMoreElements()) { IntReference<V> result = elementRefs.nextElement(); V v = result.get(); if (v != null) return v; } return null; } private V next = findNextRef(); @Override public boolean hasMoreElements() { return next != null; } @Override public V nextElement() { if (next == null) throw new NoSuchElementException(); V v = next; next = findNextRef(); return v; } }; } @Override public V putIfAbsent(int key, @NotNull V value) { IntReference<V> prev = myMap.putIfAbsent(key, createReference(key, value, myQueue)); return prev == null ? null : prev.get(); } @NotNull @Override public Collection<V> values() { Set<V> result = new THashSet<V>(); ContainerUtil.addAll(result, elements()); return result; } @Override public boolean containsValue(@NotNull V value) { for (IntEntry<IntReference<V>> entry : myMap.entries()) { if (value.equals(entry.getValue().get())) { return true; } } return false; } }