Chapter 5 - Java for Beginners Course

Final Classes and Class Members

In the same way that Java allows us to define inheritance relationships between classes, Java also provides us with a mechanism to prevent a class from being extended, to prevent a method from being overridden, or for a variable value from changing, by using the final keyword.

The effect of the final keyword depends on the context where it is being used, as follows:

final classes

We can use final in the class declaration to indicate that we don’t want to allow a class to have subclasses.

For example, let’s assume we’re building a library that holds the data of a dictionary and we want to model each entry we have, and we define a DictionaryEntry class for this.

Let’s also assume we don’t want users of our library to subclass it:

public final class DictionaryEntry {
    private String word;

    private String definition;

    public DictionaryEntry(String word, String definition) {
        this.word = word;
        this.definition = definition;
    }

    public void printEntry() {
        System.out.println(word + ": " + definition);
    }
}
To clarify, a final class can still extend from another class, but no class can extend from it.

final methods

In a similar way, if we want to allow our class to be extended but want to avoid certain methods from being overridden, you can mark those methods as final.

For example, maybe all we’re concerned about is that the behaviour of our printEntry method isn’t changed:

public class DictionaryEntry {
    private String word;

    private String definition;

    public DictionaryEntry(String word, String definition) {
        this.word = word;
        this.definition = definition;
    }

    public final void printEntry() {
        System.out.println(word + ": " + definition);
    }
}

In this case, users of our library can extend our DictionaryEntry class, but we are providing certain restrictions about the methods they can override.

final fields

A different behavior of the final keyword is when it is used with fields or variables in general. In this case, the final keyword indicates that the value of the variable can only be set once. If we try to change the value of a final variable, the compiler will show an error indicating that a value for that variable may have already been assigned.

In our DictionaryEntry class, it might not make sense to allow the word or definition to change after the object has been created, so we can mark those 2 fields as final:

private final String word;

private final String definition;
What happens if you don’t assign a value to word or definition in the constructor? Do you get a compilation error?
final classes and fields are a starting point to Immutable Objects which we won’t cover in this course.

Static final fields

A common pattern in Java is to declare constant values in an application as static final variables. These constants, by convention, are named in upper case.

For example:

public class DictionaryEntry {
    private static final String WORD_SEPARATOR = ": ";

    private final String word;

    private final String definition;

    public DictionaryEntry(String word, String definition) {
        this.word = word;
        this.definition = definition;
    }

    public final void printEntry() {
        System.out.println(word + WORD_SEPARATOR + definition);
    }
}