Conditions and condition lists
Ceylon allows one or more conditions, which together form a condition list,
in the if
, while
, and assert
statements and in if
comprehension clauses.
Note: This page is not describing the is
,
exists
, and nonempty
operators, which are considered a part of the expression grammar.
Usage
A condition list may occur in an if
, while
, or
assert
statements, or in an if
clause of a
comprehension.
Description
A condition list is satisfied if and only if every condition in the list is
satisfied. The difference between a comma-separated condition list and a single
Boolean
condition constructed using the &&
operator
is that a condition list may narrow the type of a reference, or introduce a new
reference.
Integer m(Object x) {
if (is Integer x, x < 10) {
// x is of type Integer here
return x + 1;
}
return 0;
}
Note: In the following discussion we will use if
in the code examples.
Keep in mind that condition lists may also occur in a while
or assert
statement.
Boolean conditions
A boolean condition is an expression of type Boolean
. It is satisfied
if the expression evaluates to true
.
is
conditions
A condition of form is Y x
is satisfied if the reference x
evaluates
to an instance of the type Y
. Within the associated block, x
will have
the type X&Y
, where X
is the previous type of x
.
In other words, if (is Y x)
similar to Java's if (x instanceof Y)
,
immediately followed by a typecast, (Y) x
.
Integer plusOneIfInteger(Object x) {
if (is Integer x) {
// x is of type Integer here
return = x+1;
}
return 0;
}
It's even possible to introduce a new local reference in the condition, in which case it is the new local reference which has the narrowed type:
Integer plusOneIfInteger(Object x) {
if (is Integer i=x) {
// x still of type Object here
// but i is an Integer
return i+1;
}
return 0;
}
For a variable
reference or getter you are required to use this form.
On a platform that supports concurrency this protects you from another
thread changing the value while the block protected by the if
is being
executed (and potentially violating the guarantees of the type system).
It is acceptable to specify a generic type in an is
condition, for
example, if (is List<Integer> list)
to distinguish a List<Integer>
from a List<String>
.
Note: Do not confuse the is
condition of form is Type t
with
an is
operator of form t is Type
.
exists
conditions
A condition of form exists x
means is Object x
, and is satisfied
if the reference x
refers to a non-null
value. Within the associated
block, x
will have the non-optional type X&Object
, where X
is the
previous type of x
.
In other words, if (exists x)
is similar to Java's if (x != null)
.
Integer plusOneIfExists(Integer? x) {
if (exists x) {
// x is of type Integer here
return x+1;
}
return 0;
}
It's possible to introduce a new local reference in the condition, in which case it is the new local reference that has the narrowed type:
Integer plusOneIfExists(Integer? x) {
if (exists i=x) {
// x still of type Integer? here
return i+1;
}
return 0;
}
Note: Do not confuse the exists
condition of form exists t
with an exists
operator of form t exists
.
exists
conditions with destructuring
In if
and while
statement
you can perform destructuring in an exists
condition:
[Float, Float]? point = nothing;
if (exists [x, y] = point) {
plotPoint(x,y);
}
[Float, Float]? nextPoint() => nothing;
while (exists [x, y] = nextPoint()) {
plotPoint(x,y);
}
nonempty
conditions
A condition of form nonempty xs
means is [X+] xs
, where X
is the
element type of the sequence reference x
, and is satisfied if the x
refers to a nonempty sequence. Within the associated block, x
will
have the nonempty type [X+]
.
Integer firstPlusOne(Integer[] xs) {
if (nonempty xs) {
// xs of type [Integer+] here
return xs.first+1;
}
return 0;
}
It's possible to introduce a new local reference in the condition, in which case it is the new local reference which has the narrowed type:
Integer firstPlusOne(Integer[] xs) {
if (nonempty ys=xs) {
// xs still of type Integer[] here
return ys.first+1;
}
return 0;
}
nonempty
conditions with destructuring
In if
and while
statement
you can perform destructuring in a nonempty
condition
if (nonempty [name, *rest] = process.arguments) {
print("Hello ``name``!");
}
Note: Do not confuse the nonempty
condition of form nonempty xs
with a nonempty
operator of form xs nonempty
.
See also
-
if
statement, -
while
statement, assert
statement- comprehensions
- destructuring
- Control structure conditions in the Ceylon language spec