Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Rule 1.9 defines that
onSubscribe()
on aSubscriber
should be called before calling any other method on thatSubscriber
. However, this does not clarify that ifonSubscribe()
is signalled asynchronously then there should be a happens-before relationship betweensubscribe()
and signalling ofonSubscribe()
.In absence of such a guarantee, non-final fields initialized in a constructor have no guarantee to be visible inside
onSubscribe()
.Considering a simple example:
public class UnsafeSubscriber implements Subscriber {
private boolean duplicateOnSubscribe = false;
}
If an UnsafeSubscriber instance is created in a different thread than the one that invokes onSubscribe() (true for an asynchronous Publisher), according to the java memory model, this statement inside onSubscribe():
if (duplicateOnSubscribe) {
is guaranteed to compute to false if and only if the instance is published safely between these threads. None of the rules in the specifications establish a happens-before relationship between Publisher#subscribe() and Subscriber#onSubscribe(). So, the usage above can be categorized as unsafe. In a more convoluted form, the assignment:
private boolean duplicateOnSubscribe = false;
can be interleaved with
duplicateOnSubscribe = true; such that duplicateOnSubscribe is set to false later.
Has this been considered before or am I missing something?
Fixes #486