Skip to content

Commit e703107

Browse files
committed
A Guide to @contended & False Sharing
1 parent 2420b33 commit e703107

File tree

3 files changed

+650
-0
lines changed

3 files changed

+650
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
package com.baeldung;
2+
3+
import org.openjdk.jmh.annotations.Benchmark;
4+
import org.openjdk.jmh.annotations.Scope;
5+
import org.openjdk.jmh.annotations.State;
6+
7+
@State(Scope.Benchmark)
8+
public class FalseSharing {
9+
10+
private java.util.concurrent.atomic.LongAdder builtin = new java.util.concurrent.atomic.LongAdder();
11+
private LongAdder custom = new LongAdder();
12+
13+
@Benchmark
14+
public void builtin() {
15+
builtin.increment();
16+
}
17+
18+
@Benchmark
19+
public void custom() {
20+
custom.increment();
21+
}
22+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,235 @@
1+
package com.baeldung;
2+
3+
import java.io.Serializable;
4+
import java.util.concurrent.atomic.AtomicLong;
5+
6+
/**
7+
* One or more variables that together maintain an initially zero
8+
* {@code long} sum. When updates (method {@link #add}) are contended
9+
* across threads, the set of variables may grow dynamically to reduce
10+
* contention. Method {@link #sum} (or, equivalently, {@link
11+
* #longValue}) returns the current total combined across the
12+
* variables maintaining the sum.
13+
*
14+
* <p>This class is usually preferable to {@link AtomicLong} when
15+
* multiple threads update a common sum that is used for purposes such
16+
* as collecting statistics, not for fine-grained synchronization
17+
* control. Under low update contention, the two classes have similar
18+
* characteristics. But under high contention, expected throughput of
19+
* this class is significantly higher, at the expense of higher space
20+
* consumption.
21+
*
22+
* <p>LongAdders can be used with a {@link
23+
* java.util.concurrent.ConcurrentHashMap} to maintain a scalable
24+
* frequency map (a form of histogram or multiset). For example, to
25+
* add a count to a {@code ConcurrentHashMap<String,LongAdder> freqs},
26+
* initializing if not already present, you can use {@code
27+
* freqs.computeIfAbsent(k -> new LongAdder()).increment();}
28+
*
29+
* <p>This class extends {@link Number}, but does <em>not</em> define
30+
* methods such as {@code equals}, {@code hashCode} and {@code
31+
* compareTo} because instances are expected to be mutated, and so are
32+
* not useful as collection keys.
33+
*
34+
* @since 1.8
35+
* @author Doug Lea
36+
*/
37+
public class LongAdder extends Striped64 implements Serializable {
38+
private static final long serialVersionUID = 7249069246863182397L;
39+
40+
/**
41+
* Creates a new adder with initial sum of zero.
42+
*/
43+
public LongAdder() {
44+
}
45+
46+
/**
47+
* Adds the given value.
48+
*
49+
* @param x the value to add
50+
*/
51+
public void add(long x) {
52+
Cell[] as; long b, v; int m; Cell a;
53+
if ((as = cells) != null || !casBase(b = base, b + x)) {
54+
boolean uncontended = true;
55+
if (as == null || (m = as.length - 1) < 0 ||
56+
(a = as[getProbe() & m]) == null ||
57+
!(uncontended = a.cas(v = a.value, v + x)))
58+
longAccumulate(x, null, uncontended);
59+
}
60+
}
61+
62+
/**
63+
* Equivalent to {@code add(1)}.
64+
*/
65+
public void increment() {
66+
add(1L);
67+
}
68+
69+
/**
70+
* Equivalent to {@code add(-1)}.
71+
*/
72+
public void decrement() {
73+
add(-1L);
74+
}
75+
76+
/**
77+
* Returns the current sum. The returned value is <em>NOT</em> an
78+
* atomic snapshot; invocation in the absence of concurrent
79+
* updates returns an accurate result, but concurrent updates that
80+
* occur while the sum is being calculated might not be
81+
* incorporated.
82+
*
83+
* @return the sum
84+
*/
85+
public long sum() {
86+
Cell[] as = cells; Cell a;
87+
long sum = base;
88+
if (as != null) {
89+
for (int i = 0; i < as.length; ++i) {
90+
if ((a = as[i]) != null)
91+
sum += a.value;
92+
}
93+
}
94+
return sum;
95+
}
96+
97+
/**
98+
* Resets variables maintaining the sum to zero. This method may
99+
* be a useful alternative to creating a new adder, but is only
100+
* effective if there are no concurrent updates. Because this
101+
* method is intrinsically racy, it should only be used when it is
102+
* known that no threads are concurrently updating.
103+
*/
104+
public void reset() {
105+
Cell[] as = cells; Cell a;
106+
base = 0L;
107+
if (as != null) {
108+
for (int i = 0; i < as.length; ++i) {
109+
if ((a = as[i]) != null)
110+
a.value = 0L;
111+
}
112+
}
113+
}
114+
115+
/**
116+
* Equivalent in effect to {@link #sum} followed by {@link
117+
* #reset}. This method may apply for example during quiescent
118+
* points between multithreaded computations. If there are
119+
* updates concurrent with this method, the returned value is
120+
* <em>not</em> guaranteed to be the final value occurring before
121+
* the reset.
122+
*
123+
* @return the sum
124+
*/
125+
public long sumThenReset() {
126+
Cell[] as = cells; Cell a;
127+
long sum = base;
128+
base = 0L;
129+
if (as != null) {
130+
for (int i = 0; i < as.length; ++i) {
131+
if ((a = as[i]) != null) {
132+
sum += a.value;
133+
a.value = 0L;
134+
}
135+
}
136+
}
137+
return sum;
138+
}
139+
140+
/**
141+
* Returns the String representation of the {@link #sum}.
142+
* @return the String representation of the {@link #sum}
143+
*/
144+
public String toString() {
145+
return Long.toString(sum());
146+
}
147+
148+
/**
149+
* Equivalent to {@link #sum}.
150+
*
151+
* @return the sum
152+
*/
153+
public long longValue() {
154+
return sum();
155+
}
156+
157+
/**
158+
* Returns the {@link #sum} as an {@code int} after a narrowing
159+
* primitive conversion.
160+
*/
161+
public int intValue() {
162+
return (int)sum();
163+
}
164+
165+
/**
166+
* Returns the {@link #sum} as a {@code float}
167+
* after a widening primitive conversion.
168+
*/
169+
public float floatValue() {
170+
return (float)sum();
171+
}
172+
173+
/**
174+
* Returns the {@link #sum} as a {@code double} after a widening
175+
* primitive conversion.
176+
*/
177+
public double doubleValue() {
178+
return (double)sum();
179+
}
180+
181+
/**
182+
* Serialization proxy, used to avoid reference to the non-public
183+
* Striped64 superclass in serialized forms.
184+
* @serial include
185+
*/
186+
private static class SerializationProxy implements Serializable {
187+
private static final long serialVersionUID = 7249069246863182397L;
188+
189+
/**
190+
* The current value returned by sum().
191+
* @serial
192+
*/
193+
private final long value;
194+
195+
SerializationProxy(LongAdder a) {
196+
value = a.sum();
197+
}
198+
199+
/**
200+
* Return a {@code LongAdder} object with initial state
201+
* held by this proxy.
202+
*
203+
* @return a {@code LongAdder} object with initial state
204+
* held by this proxy.
205+
*/
206+
private Object readResolve() {
207+
LongAdder a = new LongAdder();
208+
a.base = value;
209+
return a;
210+
}
211+
}
212+
213+
/**
214+
* Returns a
215+
* <a href="../../../../serialized-form.html#java.util.concurrent.atomic.LongAdder.SerializationProxy">
216+
* SerializationProxy</a>
217+
* representing the state of this instance.
218+
*
219+
* @return a {@link SerializationProxy}
220+
* representing the state of this instance
221+
*/
222+
private Object writeReplace() {
223+
return new SerializationProxy(this);
224+
}
225+
226+
/**
227+
* @param s the stream
228+
* @throws java.io.InvalidObjectException always
229+
*/
230+
private void readObject(java.io.ObjectInputStream s)
231+
throws java.io.InvalidObjectException {
232+
throw new java.io.InvalidObjectException("Proxy required");
233+
}
234+
235+
}

0 commit comments

Comments
 (0)