Skip to content

Commit c13f23d

Browse files
committed
Fixed entrySet, keySet and values to use AbstractSet & AbstractCollection iterators
1 parent a177587 commit c13f23d

6 files changed

+217
-108
lines changed

service/src/main/java/crawlercommons/urlfrontier/service/AbstractFrontierService.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -638,7 +638,7 @@ public void getURLs(GetParams request, StreamObserver<URLInfo> responseObserver)
638638
final QueueInterface currentQueue;
639639
final QueueWithinCrawl currentCrawlQueue;
640640

641-
Entry<QueueWithinCrawl, QueueInterface> e = getQueues().firsEntry();
641+
Entry<QueueWithinCrawl, QueueInterface> e = getQueues().firstEntry();
642642
currentQueue = e.getValue();
643643
currentCrawlQueue = e.getKey();
644644

service/src/main/java/crawlercommons/urlfrontier/service/ConcurrentInsertionOrderMap.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@
1515
public interface ConcurrentInsertionOrderMap<K, V> extends ConcurrentMap<K, V> {
1616

1717
/** Returns the first entry according to insertion order */
18-
Entry<K, V> firsEntry();
18+
Entry<K, V> firstEntry();
1919

2020
/** Remove & returns the first entry according to insertion order */
2121
Entry<K, V> pollFirstEntry();

service/src/main/java/crawlercommons/urlfrontier/service/ConcurrentLinkedHashMap.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,7 @@ public V put(K key, V value) {
8383
try {
8484
previous = map.put(key, value);
8585
if (previous == null) {
86-
order.add(key);
86+
order.offer(key);
8787
size.incrementAndGet();
8888
}
8989
} finally {

service/src/main/java/crawlercommons/urlfrontier/service/ConcurrentLinkedStripedMap.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -187,7 +187,7 @@ public Set<K> keySet() {
187187
/** Returns a Collection view of the values contained in this map in insertion order. */
188188
@Override
189189
public Collection<V> values() {
190-
List<V> values = new ArrayList<V>(this.size.get());
190+
List<V> values = new ArrayList<>(this.size.get());
191191
for (K key : order) {
192192
V value = map.get(key);
193193
if (value != null) {

service/src/main/java/crawlercommons/urlfrontier/service/ConcurrentOrderedMap.java

+114-74
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,17 @@
11
package crawlercommons.urlfrontier.service;
22

3-
3+
import java.util.AbstractCollection;
44
import java.util.AbstractMap;
5-
import java.util.AbstractMap.SimpleImmutableEntry;
6-
import java.util.ArrayList;
5+
import java.util.AbstractSet;
76
import java.util.Collection;
8-
import java.util.LinkedHashSet;
9-
import java.util.List;
7+
import java.util.Iterator;
108
import java.util.Map;
119
import java.util.Objects;
1210
import java.util.Set;
1311
import java.util.concurrent.ConcurrentHashMap;
1412
import java.util.concurrent.ConcurrentSkipListMap;
1513
import java.util.concurrent.atomic.AtomicLong;
1614
import java.util.concurrent.locks.StampedLock;
17-
import java.util.stream.Collectors;
1815

1916
/**
2017
* Concurrent version of LinkedHashMap Design goal is the same as for ConcurrentHashMap: Maintain
@@ -112,61 +109,77 @@ public V remove(Object key) {
112109

113110
@Override
114111
/**
115-
* Insertion order is preserved. The entry set returned is not backed up by the map.
112+
* Insertion order is preserved.
116113
*
117114
* @return a linked hash set will all keys
118115
*/
119116
public Set<K> keySet() {
120-
// Return entries in insertion order
121-
long stamp = lock.tryOptimisticRead();
122-
123-
Set<K> orderedKeys = new LinkedHashSet<>(insertionOrderMap.values());
124-
125-
// Validate the read to ensure no concurrent modifications
126-
if (!lock.validate(stamp)) {
127-
stamp = lock.readLock();
128-
try {
129-
orderedKeys = new LinkedHashSet<>(insertionOrderMap.values());
130-
} finally {
131-
lock.unlockRead(stamp);
117+
return new AbstractSet<>() {
118+
@Override
119+
public Iterator<K> iterator() {
120+
return new Iterator<>() {
121+
final Iterator<Entry<Long, K>> orderedIterator =
122+
insertionOrderMap.entrySet().iterator();
123+
124+
@Override
125+
public boolean hasNext() {
126+
return orderedIterator.hasNext();
127+
}
128+
129+
@Override
130+
public K next() {
131+
return orderedIterator.next().getValue();
132+
}
133+
134+
@Override
135+
public void remove() {
136+
throw new UnsupportedOperationException();
137+
}
138+
};
132139
}
133-
}
134140

135-
return orderedKeys;
141+
@Override
142+
public int size() {
143+
// Don't use size on CSLM as it's not in O(1) but rather O(n) and may be inacurate
144+
return valueMap.size();
145+
}
146+
};
136147
}
137148

138149
/**
139-
* Insertion order is preserved. The entry set returned is not backed up by the map.
150+
* Insertion order is preserved.
140151
*
141152
* @return a linked hash set will all entries
142153
*/
143154
@Override
144-
public Set<Entry<K, V>> entrySet() {
145-
// Return entries in insertion order
146-
long stamp = lock.tryOptimisticRead();
147-
148-
Set<Entry<K, V>> orderedEntries =
149-
insertionOrderMap.values().stream()
150-
.map(key -> new SimpleImmutableEntry<>(key, valueMap.get(key).value))
151-
.collect(Collectors.toCollection(LinkedHashSet::new));
152-
153-
// Validate the read to ensure no concurrent modifications
154-
if (!lock.validate(stamp)) {
155-
stamp = lock.readLock();
156-
try {
157-
orderedEntries =
158-
insertionOrderMap.values().stream()
159-
.map(
160-
key ->
161-
new SimpleImmutableEntry<>(
162-
key, valueMap.get(key).value))
163-
.collect(Collectors.toCollection(LinkedHashSet::new));
164-
} finally {
165-
lock.unlockRead(stamp);
155+
public Set<Map.Entry<K, V>> entrySet() {
156+
return new AbstractSet<Map.Entry<K, V>>() {
157+
@Override
158+
public Iterator<Map.Entry<K, V>> iterator() {
159+
return new Iterator<Map.Entry<K, V>>() {
160+
final Iterator<Entry<Long, K>> orderedIterator =
161+
insertionOrderMap.entrySet().iterator();
162+
163+
@Override
164+
public boolean hasNext() {
165+
return orderedIterator.hasNext();
166+
}
167+
168+
@Override
169+
public Map.Entry<K, V> next() {
170+
Entry<Long, K> nextEntry = orderedIterator.next();
171+
K key = nextEntry.getValue();
172+
V value = valueMap.get(key).value;
173+
return new AbstractMap.SimpleImmutableEntry<>(key, value);
174+
}
175+
};
166176
}
167-
}
168177

169-
return orderedEntries;
178+
@Override
179+
public int size() {
180+
return valueMap.size();
181+
}
182+
};
170183
}
171184

172185
@Override
@@ -238,14 +251,23 @@ public boolean remove(Object key, Object value) {
238251
}
239252

240253
@Override
241-
// FIXME: Should be atomic but stamped lock is not reentrant
242254
public V replace(K key, V value) {
243255

244256
long stamp = lock.writeLock();
245257
try {
246258
if (valueMap.containsKey(key)) {
247-
return put(key, value);
248-
} else return null;
259+
ValueEntry vEntry = valueMap.get(key);
260+
V oldValue = vEntry.value;
261+
vEntry.value = value;
262+
263+
return oldValue;
264+
} else {
265+
long newOrder = insertionCounter.getAndIncrement();
266+
insertionOrderMap.put(newOrder, key);
267+
valueMap.put(key, new ValueEntry(value, newOrder));
268+
269+
return null;
270+
}
249271
} finally {
250272
lock.unlockWrite(stamp);
251273
}
@@ -254,7 +276,7 @@ public V replace(K key, V value) {
254276
/*
255277
* Returns the first entry according to insertion order
256278
*/
257-
public Entry<K, V> firsEntry() {
279+
public Entry<K, V> firstEntry() {
258280
K key = insertionOrderMap.firstEntry().getValue();
259281

260282
return new AbstractMap.SimpleImmutableEntry<>(key, valueMap.get(key).value);
@@ -304,36 +326,54 @@ public boolean containsValue(Object value) {
304326

305327
@Override
306328
public void putAll(Map<? extends K, ? extends V> m) {
307-
// TODO Implement this optional operation
308-
throw new UnsupportedOperationException();
329+
330+
long stamp = lock.writeLock();
331+
try {
332+
for (Entry<? extends K, ? extends V> entry : m.entrySet()) {
333+
K key = entry.getKey();
334+
335+
// Check if key already exists
336+
ValueEntry ventry = valueMap.get(key);
337+
if (ventry != null) {
338+
ventry.value = entry.getValue();
339+
} else {
340+
long newOrder = insertionCounter.getAndIncrement();
341+
insertionOrderMap.put(newOrder, key);
342+
valueMap.put(key, new ValueEntry(entry.getValue(), newOrder));
343+
}
344+
}
345+
} finally {
346+
lock.unlock(stamp);
347+
}
309348
}
310349

311350
/** Returns a Collection view of the values contained in this map in insertion order. */
312351
@Override
313352
public Collection<V> values() {
314-
List<V> values;
315-
316-
// Return entries in insertion order
317-
long stamp = lock.tryOptimisticRead();
318-
319-
values =
320-
insertionOrderMap.values().stream()
321-
.map(key -> valueMap.get(key).value)
322-
.collect(Collectors.toCollection(ArrayList::new));
323-
324-
// Validate the read to ensure no concurrent modifications
325-
if (!lock.validate(stamp)) {
326-
stamp = lock.readLock();
327-
try {
328-
values =
329-
insertionOrderMap.values().stream()
330-
.map(key -> valueMap.get(key).value)
331-
.collect(Collectors.toCollection(ArrayList::new));
332-
} finally {
333-
lock.unlockRead(stamp);
353+
return new AbstractCollection<>() {
354+
@Override
355+
public Iterator<V> iterator() {
356+
return new Iterator<>() {
357+
final Iterator<Entry<Long, K>> orderedIterator =
358+
insertionOrderMap.entrySet().iterator();
359+
360+
@Override
361+
public boolean hasNext() {
362+
return orderedIterator.hasNext();
363+
}
364+
365+
@Override
366+
public V next() {
367+
K key = orderedIterator.next().getValue();
368+
return valueMap.get(key).value;
369+
}
370+
};
334371
}
335-
}
336372

337-
return values;
373+
@Override
374+
public int size() {
375+
return valueMap.size();
376+
}
377+
};
338378
}
339379
}

0 commit comments

Comments
 (0)