Coding Guidelines

1. Literature

Must read:

The best way to write secure and reliable applications:

2. Immutability

Don’t declare objects with mutable state

All instance variables in a class should be declared as private final. This will make the class immutable and its instance members hidden from the outside.

Why immutable objects?

  • they are simpler to construct, test, and reason about

  • truly immutable objects are always thread-safe

  • they help to avoid temporal coupling

  • their usage is side-effect free

  • they prevent NULL references

  • immutable classes tend to be smaller

How do we modify a state of an immutable object? We create a new object with a different state.

Note: There are different gradients of immutability

2.1. When to Use Mutable Objects

Start with objects as immutable and only relax the immutability if absolutely necessary. Don’t use getters/setters IDE feature automatically, think about how the class is used and only provide the mutability that is absolutely necessary.

Few examples of when to use mutable classes:

  • for data deserialization (dto objects). Deserialization libraries often require setter methods on deserialized objects.

  • for complex objects where creating new objects comes at a cost of degradation in performance and maintainability, for example tree like structures

3. Program to an Interface

Don’t inject concrete classes to objects

Classes should communicate with each other through interfaces/contracts. This is particulary important in dependency injection where we always want to pass an interface through a constructor and not a concrete class.

Why interfaces?

3.1. When to inject concrete class

It is ok to inject concrete classes for low level implementation design that usualy involves methods that process data in a side-effect free way. For example, string processing, data parsing etc…​

4. Instanceof and Type Casting

Don’t use instanceof and type castings

These are code smells, use polymorphism instead.

Note: It’s not entirely possible to avoid them, they are normally used for exception handling, annotation processing, and in integration with third party libraries that relies on the generic Object class.

5. Utility Classes

Think twice before creating a utility class

Utility Class, also known as Helper class, is a class, which contains only static methods, it is stateless, and cannot be instantiated. It contains a bunch of related methods, so they can be reused across the application. As an example consider Apache StringUtils, CollectionUtils or java.lang.Math.

5.1. Problem
  • utility classes introduce tight coupling between classes

  • utility classes often break single-responsibility principle, they tend to accumulate more and more code which may not be related

  • it’s harder to test a class that depends on an utility class

5.2. Example

Common mistake is to create utility class that relies on some external dependency such as database connection, http client, etc…​

DbUtils.insertRecord(record, connection);

This way all the classes that use DbUtils are tightly coupled to connection class.

It’s better to create a new database object using connection as argument in the constructor. That way connection class will be hidden inside the database object:

new Database(connection).insertRecord(record);
5.3. When It’s Safe to Use Utility Classes

You can use them for low level implementation design that involves methods that process data in a side-effect free way. For example, string processing, data parsing etc…​

6. Null Handling

Avoid returning NULL

Why?:

  • NullPointerException

  • ad-hoc error handling

  • slow failing

Altertnatives to returning null:

  • Return a neutral value (empty string, empty collection, empty map, 0 value…​)

  • Return value wrapped in Optional class

  • Return a default value

  • Throw an exception

Avoid passing NULL

Unless you are working with an API which expects you to pass null, you should avoid passing null in your code whenever possible. You can either focus your efforts on checking for null or not ever passing null. The more elegant solution is to focus on never passing null. By doing this you will end up writting less code and avoid decisions about how to handle null inside a method that doesn’t have enough context to decide what to do.

6.1. Example

One very common example for passing null is to create a method which determines its result based on whether an argument is NULL or not.

int foo(int bar, Integer multiplier) {
	if (multiplier == null) return bar * 2;
	return bar * multiplier;
}

Usage of this function allows passing NULL parameter, for instance foo(4, null);

It’s better to separate foo method into two overloaded functions and change Integer to a primitive int to forbid NULL values.

foo(int bar) {}
foo (int bar, int multiplier) {}

7. Class Naming Conventions

Don’t use -er suffixes

Don’t use -Impl suffix

naming
naming ext

Avoid ER ending names. Name classes by what they represent, not by what they do. For example, instead of ApplicationRunner use Application.run(), instead of ConfigurationLoader use Configuration.load().

Impl suffix is often used when only one implementation of a specific interface exists. For instance, the implementation of an interface named PatientDao would be PatientDaoImpl. We don’t need Impl suffix to remind us that the class implements an interface, that is completly redundant. It would be better to reveal an implementation detail in the name, for example, SqlPatientDao or PostgresPatientDao. We can go even further and remove dao suffix so the end result can look like SqlPatients - pretty clean, right?

8. Comments in Code

Don’t explain yourself using comments in code

//check if user can read a resource
if (user.permissions().canRead(id) || user.roles().contains("admin"))

Use functions instead:

if (userCanRead(id))

Use comments when you need to explain your intent, to clarify something, or to warn a developer:

// This is a very bad hack. It is introduced to overcome class loader problem with Karaf.
ClassLoader cl = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(null);

8.1. Javadoc

Class and interface declarations as well as interface method declarations should always contain javadoc.

Public and private methods should be commented when additional explanation is needed.

It is also a good practice to add comments to complex variables such as regex patterns.

8.2. Reference

9. Class Inheritance in Tests

Don’t use class inheritence in test suites as a way of code reuse.

Don’t share test data between tests through class inheritance.

Inheritance is the mechanism in Java by which one class is allowed to inherit the features(fields and methods) of another class

Super Class: The class whose features are inherited (super class, base class, or parent class).

Sub Class: The class that inherits the other class(sub class, derived class, extended class, or child class). The subclass can add its own fields and methods in addition to the superclass fields and methods.

Reusability: Inheritance supports the concept of “reusability”, i.e. when we want to create a new class and there is already a class that includes some of the code that we want, we can derive our new class from the existing class.

9.1. Problem

  • God object and Non-reusable code

    • Let’s say a test class needs to access a database in order to validate behaviour of some part of an application. To share this functionality with another test class, the code is moved to an abstract class to make it accessible from the outside. Methods declared in an abstract class lack appropriate context. They are defined in an artificial class that is nothing but a dummy bag of functions, very similar to utility classes. The main problem is maintainability because such class can quickly turn into a God object, a well known anti-pattern. The second problem is reusability, it is very hard to reuse methods in abstract class in other test suites.

    • Solution: favor composition over inheritance. Whenever you need to reuse a piece of code, create appropriate class/service that will encapsulate this functionality and inject it into a test class.

  • Tests are not isolated

    • By sharing test data in abstract classes through protected instance variables we make tests less isolated. Any change to one those variables affects multiple tests. These variables are basically global variables and should be avoided as much as possible.

    • Solution: Try to define a different set of test data for every test. If you need to share some data do it through private methods or through separate classes.

9.2. When to Use Inheritance

Use abstract classes when you need to inject external dependencies/libraries, business objects or other services in tests. These objects can be instantiated/composed inside an abstract class so that each derived test class can access them. Abstract class is also a good place to define setup methods, pieces of code that needs to be triggered before test execution.

10. Test Naming

Test names should be as descriptive as possible

Test name should express a specific requirement. This requirement should be derived from either a business or a technincal requirement. Unit test case represents a small piece of that requirement.

Here is an example of a test method, that verifies successful patient creation, but does not reveal enough information in it’s description.

public void testCreate() { ... }

Use this naming alternatives instead:

createsPatient() { ... }

shouldCreatePatient() { ... }

testPatientCreation() { ... }

11. Test Assertions

Reduce number of assertions per test

The more assertions a test has, the less readable and reusable it becomes. Ideally a test should have only one assert statement. This is often not possible, however we should strive towards this goal. In order to achieve this, tests should focus more on matching objects - not variables. This way of thinking can significantly reduce the amount of duplicated code and increase reusability.

Here is an example using junit for matching variables vs Hamcrest library for matching objects:

@Test
public void retrievesEntityFromDatabase() {
	Patient patient = patients.get("patientId");
	assertEquals(patient.id(), "patientId");
	assertEquals(patient.name(), "patientName");
	assertEquals(patient.nurse(), "nurseName");
	assertTrue(patient.isActive());
}

Using Hamcrest:

@Test
public void retrievesPatientFromDatabase() {
	Patient patient = patients.get("patientId");
	assertThat(
		patient,
		new PatientEqualTo(patient.builder()
			.id("patientId")
			.name("patientName")
			.nurse("nurseName")
			.isActive(true)
		)
	);
}

Now imagine you have to assert patient data in multiple tests. In the first example you will have to write assert statements for each test while using Hamcrest only once.

Here is the complete example

12. Fake Objects

Prefer fakes over mocks

Problems with mockito code:

  • it is hard to reason about

  • it can become pretty complex very fast

  • its very hard to reuse mocked code

  • tests that rely on mocks are inherently coupled to the implementation of the system and are fragile as the result

Instead of mockito use fake objects. Fake object represents a simple/fake implementation of an interface used to accomodate testing scenarios.

Checkout this example of using fakes vs mockito.

Use mock framework only when the cost of creating and maintaining a fake object is higher than using the framework. It is also advised to use mocks when we don’t have direct access to a 3pp source code and if interface we are trying to test is complex.