Class modifiers control how a class or mixin can be used, both from within its own library, and from outside of the library where it’s defined.
Modifier keywords come before a class or mixin declaration. For example, writing abstract class
defines an abstract class. The full set of modifiers that can appear before a class declaration include:
abstract
base
final
interface
sealed
mixin
Abstract
To define a class that doesn’t require a full, concrete implementation of its entire interface, use the abstract
modifier.
Abstract classes cannot be constructed from any library, whether its own or an outside library. Abstract classes often have abstract methods.
// Library a.dart
abstract class Vehicle {
void moveForward(int meters);
}
// Library b.dart
import 'a.dart';
// Error: Cannot be constructed
Vehicle myVehicle = Vehicle();
// Can be extended
class Car extends Vehicle {
int passengers = 4;
// ···
}
// Can be implemented
class MockVehicle implements Vehicle {
@override
void moveForward(int meters) {
// ...
}
}
base
To enforce inheritance of a class or mixin’s implementation, use the base
modifier. A base class disallows implementation outside of its own library. This guarantees:
The base class constructor is called whenever an instance of a subtype of the class is created.
All implemented private members exist in subtypes.
A new implemented member in a
base
class does not break subtypes, since all subtypes inherit the new member.- This is true unless the subtype already declares a member with the same name and an incompatible signature.
You must mark any class which implements or extends a base class as base
, final
, or sealed
. This prevents outside libraries from breaking the base class guarantees.
// Library a.dart
base class Vehicle {
void moveForward(int meters) {
// ...
}
}
// Library b.dart
import 'a.dart';
// Can be constructed
Vehicle myVehicle = Vehicle();
// Can be extended
base class Car extends Vehicle {
int passengers = 4;
// ...
}
// ERROR: Cannot be implemented
base class MockVehicle implements Vehicle {
@override
void moveForward() {
// ...
}
}
interface
To define an interface, use the interface
modifier. Libraries outside of the interface’s own defining library can implement the interface, but not extend it. This guarantees:
When one of the class’s instance methods calls another instance method on
this
, it will always invoke a known implementation of the method from the same library.Other libraries can’t override methods that the interface class’s own methods might later call in unexpected ways. This reduces the fragile base class problem.
// Library a.dart
interface class Vehicle {
void moveForward(int meters) {
// ...
}
}
// Library b.dart
import 'a.dart';
// Can be constructed
Vehicle myVehicle = Vehicle();
// ERROR: Cannot be inherited
class Car extends Vehicle {
int passengers = 4;
// ...
}
// Can be implemented
class MockVehicle implements Vehicle {
@override
void moveForward(int meters) {
// ...
}
}
abstract interface
The most common use for the interface
modifier is to define a pure interface. Combine the interface
and abstract
modifiers for an abstract interface class
.
Like an interface
class, other libraries can implement, but cannot inherit, a pure interface. Like an abstract
class, a pure interface can have abstract members.
final
To close the type hierarchy, use the final
modifier. This prevents subtyping from a class outside of the current library. Disallowing both inheritance and implementation prevents subtyping entirely. This guarantees:
You can safely add incremental changes to the API.
You can call instance methods knowing that they haven’t been overwritten in a third-party subclass.
Final classes can be extended or implemented within the same library. The final
modifier encompasses the effects of base
, and therefore any subclasses must also be marked base
, final
, or sealed
.
// Library a.dart
final class Vehicle {
void moveForward(int meters) {
// ...
}
}
// Library b.dart
import 'a.dart';
// Can be constructed
Vehicle myVehicle = Vehicle();
// ERROR: Cannot be inherited
class Car extends Vehicle {
int passengers = 4;
// ...
}
class MockVehicle implements Vehicle {
// ERROR: Cannot be implemented
@override
void moveForward(int meters) {
// ...
}
}
sealed
To create a known, enumerable set of subtypes, use the sealed
modifier. This allows you to create a switch over those subtypes that is statically ensured to be exhaustive.
The sealed
modifier prevents a class from being extended or implemented outside its own library. Sealed classes are implicitly abstract.
They cannot be constructed themselves.
They can have factory constructors.
They can define constructors for their subclasses to use.
Subclasses of sealed classes are, however, not implicitly abstract.
The compiler is aware of any possible direct subtypes because they can only exist in the same library. This allows the compiler to alert you when a switch does not exhaustively handle all possible subtypes in its cases:
sealed class Vehicle {}
class Car extends Vehicle {}
class Truck implements Vehicle {}
class Bicycle extends Vehicle {}
// ERROR: Cannot be instantiated
Vehicle myVehicle = Vehicle();
// Subclasses can be instantiated
Vehicle myCar = Car();
String getVehicleSound(Vehicle vehicle) {
// ERROR: The switch is missing the Bicycle subtype or a default case.
return switch (vehicle) {
Car() => 'vroom',
Truck() => 'VROOOOMM',
};
}
If you don’t want exhaustive switching, or want to be able to add subtypes later without breaking the API, use the final
modifier.