Synchronized blocks in value constructors

João Mendonça jf.mend at gmail.com
Mon Sep 25 00:57:35 UTC 2023


I would like to propose a new syntax to specify that, in a value class,
some fields must be assigned atomically together:

*** synchronized blocks in value class constructors. ***


Advantages:

    - granularity - only the specific fields involved in an invariant need
to be assigned in the same synchronized block. Multiple blocks can be used
for independent invariants. Tearing is allowed between blocks and between
fields assigned outside blocks. VMs could take advantage of this to perform
optimizations.
    - economy - no new keyword/annotation/interface needed.
    - compatibility - synchronized blocks are currently illegal in
constructors.
    - safety - just like with identity classes, the absence of a
synchronized block in a constructor means that the whole constructor is
synchronized, i.e. all fields are written atomically.
    - safety - an empty synchronized block is required to indicate that
tearing is allowed between any fields.
    - clarity - "A synchronized block of field assignments" is a very
intuitive description of the semantics involved, given the meaning of the
word "synchronized" in english.


Disadvantages:

    - aesthetics - an empty synchronized block is required to indicate that
tearing is allowed between any fields.
    - safety - a user of a value class has no automatic way to be informed
of if/where tearing may occur (can be fixed with an update to the
generation of java docs).
    - clarity - synchronized blocks (§14.18.) have two meanings:
        - The old meaning in regular methods: only one thread may be
running inside of a block for the object given in the expression clause
        - The new meaning in constructors: all assignments in a block are
written atomically (no expression clause)

Example:

    value class Range {
        long start, end;
        public Range(long start, long end) {
            if (start > end) throw new IllegalArgumentException();
            synchronized {
                this.start = start;
                this.end = end;
            }
        }    }

Having all fields assigned in the same synchronized block, as above,
is equivalent to declaring no synchronized blocks:

    value class Range {
        long start, end;

        public Range(long start, long end) {
            if (start > end) throw new IllegalArgumentException();
            this.start = start;
            this.end = end;
        }    }


Another example:

    value class Point {
        double x, y;
                public Point(double x, double y) {
            synchronized {}
            this.x = x;            this.y = y;
        }    }



João Mendonça
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://mail.openjdk.org/pipermail/valhalla-spec-observers/attachments/20230925/38edbe5c/attachment.htm>


More information about the valhalla-spec-observers mailing list