Annotation Type ThreadSafe
-
@Target(TYPE) @Retention(RUNTIME) @Inherited @Documented public @interface ThreadSafeThis annotation indicates that the class/interface it is applied to is thread safeAn object is thread safe if no sequences of accesses (like reads and writes to public fields or calls to public methods) may put the object into an invalid state, or cause it to violate its contract, regardless of the interleaving of those actions at runtime.
This annotation has two related-but-distinct purposes:
- For humans: it indicates that the class/interface (and subclasses) is thread-safe
- For machines: it causes the annotated class/interface -- and all of its subtypes -- to be
validated by the
com.google.errorprone.bugpatterns.threadsafety.ThreadSafeCheckerBugChecker.
ThreadSafeCheckeris neither necessary nor sufficient to guarantee the thread safety of a class. In fact, it is not possible to determine thread safety through static code analysis alone, and the goal ofThreadSafeCheckeris to steer the code towards using standard thread-safe patterns, and then to assist the developer in avoiding common mistakes. It is not meant as a substitute for diligent code review by a knowledgeable developer.Also note that the only easy way to guarantee thread-safety of a class is to make it immutable, and you should do that whenever possible (or at the least, make as much of the class be immutable). Otherwise, writing a thread-safe class is inherently tricky and error prone, and keeping it thread-safe is even more so.
The remainder of this javadoc describes the heuristics enforced by
ThreadSafeCheckerand the relatedcom.google.errorprone.bugpatterns.threadsafety.GuardedByCheckerandcom.google.errorprone.bugpatterns.threadsafety.ImmutableCheckeron which the former relies.The
ThreadSafeCheckerheuristics enforce that every field meets at least one of these requirements:- It is both
finaland its type is deemed inherently deeply thread-safe; and/or - it is annotated with either
GuardedBy(some other annotations namedGuardedByalso work, though this the preferred);
GuardedBy.A type is deemed inherently deeply thread-safe if it meets two requirements. The first requirement is that it meets at least one of these four conditions:
- it is listed as a well-known immutable type in
com.google.errorprone.bugpatterns.threadsafety.WellKnownMutability(e.g. a field of typeString); and/or - it is listed as a well-known thread-safe type in
com.google.errorprone.bugpatterns.threadsafety.WellKnownThreadSafety(e.g. a field of typeAtomicBoolean); and/or - it is annotated with
Immutable; and/or - it is annotated with
ThreadSafe.
This first requirement means the type is at least inherently shallowly thread-safe.
Fields annotated with
javax.annotation.concurrent.GuardedByare likely the meat of a mutable thread-safe class: these are things that need to be mutated, but should be done so in a safe manner -- i.e., (most likely) in critical sections of code that protect their access by means of a lock. See more information in that annotation's javadoc.As stated before, the heuristics above are not sufficient to guarantee the thread safety of a class. Also as stated before, thread-safety is tricky, and requires diligent analysis by skilled people. That said, we provide here a few examples of common examples of ways to break these heuristics, so as to help you avoid them:
- a non-private
@GuardedByfield -- i.e.if a non-private@GuardedByfield is accessed outside the class, the code that enforces@GuardedBywill not prevent unprotected access and/or modifications to the field; - indirect access to the field. There are several ways in which code may access the objects
stored in the field indirectly (i.e. not directly referencing the field). In all these
cases,
@GuardedByoffers no enforcement. Here's some examples:- if the
@GuardedByfield instance is part of an object by a method (e.g. a simple getter method or constructor parameter); - if a method takes an out-parameter and the method calls a method in that
out-parameter passing the instance of the
@GuardedByfield, and that instance is stored in the out-parameter (i.e. a simple setter method);
- if the
- methods that perform multiple operations -- e.g., if a class
Foocontains aAtomicInteger(which is an inherently deeply thread-safe data structure), the following code makes this class not thread-safe:private void incrementMyAtomicInteger() { myAtomicInteger.set(myAtomicInteger.get() + 1); }