Note: information on this page refers to Ceylon 1.2, not to the current release.
Classes
This is the second step in our tour of the Ceylon language. In the previous leg you learned some of the basics of the syntax of Ceylon. In this leg we're going to learn how to define classes with methods and attributes.
Identifier naming
The case of the first character of an identifier is significant. Type (interface, class, and type parameter) names must start with an initial capital letter. Function and value names start with an initial lowercase letter or underscore. The Ceylon compiler is very fussy about this. You'll get a compilation error if you write:
class hello() { ... } //compile error
or:
String Name = .... //compile error
There is a way to work around this restriction, which is
mainly useful when calling legacy Java code. You can "force"
the compiler to understand that an identifier is a type name
by prefixing it with \I
, or that it is a function or value
name by prefixing it with \i
. For example, \iRED
is
considered an initial lowercase identifier.
So the following declarations are acceptable, but definitely not recommended, except in the interop scenario:
class \Ihello() { ... } //OK, but not recommended
and:
String \iName = .... //OK, but not recommended
Creating your own class
Our first class is going to represent points in a polar coordinate system. Our class has two parameters, two methods, and an attribute.
"A polar coordinate"
class Polar(Float angle, Float radius) {
shared Polar rotate(Float rotation)
=> Polar(angle+rotation, radius);
shared Polar dilate(Float dilation)
=> Polar(angle, radius*dilation);
shared String description
= "(``radius``,``angle``)";
}
There's two things in particular to notice here:
The parameters used to instantiate a class are specified as part of the class declaration, right after the name of the class. This syntax is less verbose and more regular than Java, C#, or C++. We do have constructors in Ceylon, but we rarely need them, and they shouldn't be the first thing you reach for.
We make use of the parameters of a class anywhere within the body of the class. In Ceylon, we often don't need to define explicit members of the class to hold the parameter values. Instead, we can access the parameters
angle
andradius
directly from therotate()
anddilate()
methods, and from the expression which specifies the value ofdescription
.
Notice also that Ceylon doesn't have a new
keyword to indicate instantiation,
we just "invoke the class", writing Polar(angle, radius)
.
Hiding implementation details
Ceylon doesn't make a distinction between public
, protected
and "default"
visibility like Java does;
here's why.
Instead, the language distinguishes between:
- program elements which are visible only inside the scope in which they are defined, and
- program elements which are visible wherever the thing they belong to (a type, package, or module) is visible.
By default, members of a class are hidden from code outside the body of the
class. By annotating a member with the shared
annotation, we declare that
the member is visible to any code to which the class itself is visible.
And, of course, a class itself may be hidden from other code. By default, a
toplevel class is hidden from code outside the package in which the class is
defined. Annotating a top level class with shared
makes it visible to any
code to which the package containing the class is visible.
Got the idea? We're playing Russian dolls here.
Exposing parameters as attributes
If we want to expose the angle
and radius
of our Polar
coordinate to
other code, we need to define attributes of the class. It's very common to
assign parameters of a class directly to a shared
attribute of the class,
so Ceylon provides a streamlined syntax for this.
"A polar coordinate"
class Polar(angle, radius) {
shared Float angle;
shared Float radius;
shared Polar rotate(Float rotation)
=> Polar(angle+rotation, radius);
shared Polar dilate(Float dilation)
=> Polar(angle, radius*dilation);
shared String description
= "(``radius``,``angle``)";
}
Code that uses Polar
can access the attributes of the class using a very
convenient syntax.
Cartesian cartesian(Polar polar)
=> Cartesian(polar.radius*cos(polar.angle),
polar.radius*sin(polar.angle));
There's an even more compact way to write the code above, though it's often less readable:
"A polar coordinate"
class Polar(shared Float angle, shared Float radius) {
shared Polar rotate(Float rotation)
=> Polar(angle+rotation, radius);
shared Polar dilate(Float dilation)
=> Polar(angle, radius*dilation);
shared String description
= "(``radius``,``angle``)";
}
This illustrates an important feature of Ceylon: there is almost no essential difference, aside from syntax, between a parameter of a class, and a value declared in the body of the class.
Instead of declaring the attributes in the body of the class, we simply
annotated the parameters shared
. We encourage you to avoid this shortcut
when you have more than one or two parameters.
Initializing attributes
The attributes angle
and radius
are references, the closest thing
Ceylon has to a Java field. Usually we specify the value of a reference
when we declare it.
Float x = radius * sin(angle);
String greeting = "Hello, ``name``!";
Integer months = years * 12;
On the other hand, it's sometimes useful to separate declaration from assignment.
shared String description;
if (exists label) {
description = label;
}
else {
description = "(``radius``,``angle``)";
}
But if there's no constructors in Ceylon, where precisely should we put this code? We put it directly in the body of the class!
"A polar coordinate with an optional label"
class Polar(angle, radius, String? label) {
shared Float angle;
shared Float radius;
shared String description;
if (exists label) {
description = label;
}
else {
description = "(``radius``,``angle``)";
}
// ...
}
The Ceylon compiler forces you to specify a value of any reference before making use of the reference in an expression.
Integer count;
void inc() {
count++; //compile error
}
We'll learn more about this later in the tour.
Abstracting state using attributes
If you're used to writing JavaBeans, you can think of a reference as a combination of several things:
- a field,
- a getter, and, sometimes,
- a setter.
That's because not every value is a reference like the one we've just seen; others are more like a getter method, or, sometimes, like a getter and setter method pair.
We'll need to expose the equivalent cartesian coordinates of a Polar
.
Since the cartesian coordinates can be computed from the polar coordinates,
we don't need to define state-holding references. Instead, we can define
the attributes as getters.
import ceylon.math.float { sin, cos }
"A polar coordinate"
class Polar(angle, radius) {
shared Float angle;
shared Float radius;
shared Float x => radius * cos(angle);
shared Float y => radius * sin(angle);
// ...
}
Notice that the syntax of a getter declaration looks a lot like a method declaration with no parameter list.
So in what way are attributes "abstracting state"? Well, code that uses
Polar
never needs to know if an attribute is a reference or a getter.
Now that we know about getters, we could rewrite our description
attribute as a getter, without affecting any code that uses it.
"A polar coordinate, with an optional label"
class Polar(angle, radius, String? label) {
shared Float angle;
shared Float radius;
shared String description {
if (exists label) {
return label;
}
else {
return "(``radius``,``angle``)";
}
}
}
Living without static members
Right at the beginning of the tour, we mentioned that Ceylon doesn't have
static
members like in Java, C#, or C++. Instead of a static
member,
we either:
- use a toplevel function or value declaration, or
- in the case where several "static" declarations need to share some private
stuff, members of a singleton
object
declaration, which we'll meet right at the start of the next chapter.
The lack of static members results in a minor gotcha for newcomers.
Gotcha!
The syntax Polar.radius
is legal in Ceylon, and we even call it a
static reference,
but it does not usually mean what you think it means!
Sure, if you're taking advantage of Ceylon's Java interop, you can call a static member of a Java class using this syntax, just like you would in Java:
import java.lang { Runtime }
Integer procs = Runtime.runtime.availableProcessors();
Or, alternatively, you could write the following, directly import
ing
the static member:
import java.lang { Runtime { runtime } }
Integer procs = runtime.availableProcessors();
But in regular Ceylon code, an expression like Polar.radius
is not a
reference to a static member of the class Polar
. We'll come back to the
question of what a "static reference" really is, when we discuss
higher-order functions.
Living without overloading
It's time for some bad news: Ceylon doesn't have method or constructor overloading (the truth is that overloading is the source of various problems in Java, especially when generics come into play). However we can emulate most non-harmful uses of constructor or method overloading using:
- defaulted parameters,
- variadic parameters (varargs), and
- union types or enumerated type constraints.
We're not going to get into all the details of these workarounds right now, but here's a quick example of each of the three techniques:
//defaulted parameter
void println(String line, String eol = "\n")
=> process.write(line + eol);
//variadic parameter
void printlns(String* lines) {
for (line in lines) {
println(line);
}
}
//union type
void printName(String|Named name) {
switch (name)
case (is String) {
println(name);
}
case (is Named) {
println(name.first + " " + name.last);
}
}
Don't worry if you don't completely understand the third example just yet, we'll come back to it later in the tour.
Let's make use of this idea to "overload" the "constructor" of Polar
.
"A polar coordinate with an optional label"
class Polar(angle, radius, String? label=null) {
shared Float angle;
shared Float radius;
shared String description {
if (exists label) {
return label;
}
else {
return "(``radius``,``angle``)";
}
}
// ...
}
Now we can create Polar
coordinates with or without labels:
Polar origin = Polar(0.0, 0.0, "origin");
Polar coord = Polar(r, theta);
Later, we'll learn about named arguments, which we often use to make instantiation expressions more readable, especially when the class has more than two parameters:
Polar origin = Polar { angle = 0.0; radius = 0.0; label = "origin"; };
Polar coord = Polar { radius = r; angle = theta; };
Gotcha!
Even with these "emulation" techniques, not every case of a legal overloaded Java method can be represented directly in Ceylon. In such situations it's necessary to exert a little more effort to come up with distinct names.
There's more...
In the next chapter, we'll continue our investigation of attributes, and especially variable attributes. We'll also meet Ceylon's control structures.
(We'll wait until a later chapter to learn more about methods.)