DEV Community

Bellamer
Bellamer

Posted on

Supercharging Java with Project Lombok: Simplify Your Code, Boost Your Productivity

Java is known for its verbosity, especially when dealing with repetitive boilerplate code like getters, setters, constructors, and toString methods. While necessary, this clutter can slow down development and make code harder to read and maintain. Project Lombok steps in to solve this problem by automatically generating boilerplate code at compile time.

In this guide, we’ll dive deep into why Lombok is a must-have tool in Java, how to set it up, and take a look behind the scenes to understand how Lombok uses annotation processing to map and modify Java’s Abstract Syntax Tree (AST), making sure our code is concise without sacrificing functionality.

Why Do We Need Lombok?

Java requires a significant amount of boilerplate code. Consider a simple POJO that includes fields, a constructor, getters, setters, and a toString method:

public class User {
    private String name;
    private int age;

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{name='" + name + "', age=" + age + '}';
    }
}
Enter fullscreen mode Exit fullscreen mode

This is manageable for a small project, but as applications scale, the boilerplate can quickly become overwhelming.

The Solution: Lombok

With Lombok, we can avoid all this repetitive code. Here’s how the same class looks with Lombok:

import lombok.Data;

@Data
public class User {
    private String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode

With just one annotation, Lombok generates the getters, setters, toString, equals, and hashCode methods, making the code cleaner and easier to maintain.

How to Set Up Lombok

Step 1: Adding Lombok Dependency

To use Lombok in a Maven project, add this dependency to pom.xml:

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.28</version>
    <scope>provided</scope>
</dependency>
Enter fullscreen mode Exit fullscreen mode

For Gradle:

compileOnly 'org.projectlombok:lombok:1.18.28'
annotationProcessor 'org.projectlombok:lombok:1.18.28'
Enter fullscreen mode Exit fullscreen mode

Step 2: IDE Setup

Make sure your IDE has the Lombok plugin installed. In IntelliJ IDEA:

1.  Go to Settings > Plugins.
2.  Search for Lombok.
3.  Install the Lombok plugin.
4.  Enable annotation processing under Settings > Build, Execution, Deployment > Compiler > Annotation Processors.
Enter fullscreen mode Exit fullscreen mode

How Lombok Works: Behind the Scenes

The Power of Annotation Processing

Lombok uses Java Annotation Processors to interact with the Java Abstract Syntax Tree (AST). Annotation processors analyze and potentially modify the code’s structure during the compilation process. Lombok’s annotation processor leverages this to generate methods like getters, setters, and toString, among others, by inserting them directly into the AST before the code is compiled.

Understanding the Abstract Syntax Tree (AST)

The AST is an internal representation of the code, breaking down the source into a structured tree that the compiler can process. When you write Java code, each element (like classes, fields, methods) is mapped to a node in the AST.

When Lombok’s annotation processor encounters a class with Lombok annotations, it modifies the AST nodes to add methods directly into the tree. This means the boilerplate methods are generated during compilation and are not part of the source code. By the time the code compiles, it has been augmented with all necessary methods, which is why they work seamlessly.

Timing: When and How Lombok Generates Code

Lombok’s annotations are processed during the compilation phase, between parsing the Java source code and generating bytecode. Here’s the step-by-step process:

1.  AST Creation: The compiler reads the Java source code and generates the AST.
2.  Annotation Processing: Lombok’s annotation processor scans the AST for Lombok annotations.
3.  AST Manipulation: When an annotation like @Getter or @Setter is found, Lombok adds nodes to the AST to represent the corresponding methods.
4.  Bytecode Generation: The modified AST is used to generate the final bytecode, which includes the methods added by Lombok.
Enter fullscreen mode Exit fullscreen mode

As a result, when you run your code, the getters, setters, and other methods appear as if they were part of the original source code, even though they were added at compile time.

Lombok Annotations Explained

  • @Getter and @Setter

These annotations generate getter and setter methods for your fields, effectively saving you the effort of writing them yourself.

import lombok.Getter;
import lombok.Setter;

@Getter
@Setter
public class User {
    private String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode
  • @ToString

@ToString generates a toString method that includes all fields by default, with the option to exclude specific fields.

import lombok.ToString;

@ToString(exclude = password)
public class User {
    private String name;
    private int age;
    private String password;
}
Enter fullscreen mode Exit fullscreen mode
  • @EqualsAndHashCode

Generates equals and hashCode methods. These methods are essential for comparing objects and using them in collections like HashMap or HashSet.

import lombok.EqualsAndHashCode;

@EqualsAndHashCode
public class User {
    private String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode
  • @NoArgsConstructor, @AllArgsConstructor, and @RequiredArgsConstructor

These annotations generate constructors with different parameter options.

• @NoArgsConstructor: No-argument constructor.
• @AllArgsConstructor: Constructor with parameters for all fields.
• @RequiredArgsConstructor: Constructor for fields marked final or @NonNull.
Enter fullscreen mode Exit fullscreen mode
import lombok.NoArgsConstructor;
import lombok.AllArgsConstructor;
import lombok.RequiredArgsConstructor;

@NoArgsConstructor
@AllArgsConstructor
@RequiredArgsConstructor
public class User {
    private final String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode
  • @Data

Combines @Getter, @Setter, @ToString, @EqualsAndHashCode, and @RequiredArgsConstructor.

import lombok.Data;

@Data
public class User {
    private String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode
  • @Builder

Implements the builder pattern, which is especially useful for constructing complex objects with multiple parameters.

import lombok.Builder;

@Builder
public class User {
    private String name;
    private int age;
}
Enter fullscreen mode Exit fullscreen mode

Usage:

User user = User.builder()
    .name("John Doe")
    .age(30)
    .build();
Enter fullscreen mode Exit fullscreen mode
  • @Slf4j (Logging)

Provides an SLF4J Logger instance, simplifying logging setup.

import lombok.extern.slf4j.Slf4j;

@Slf4j
public class UserService {

    public void createUser() {
        log.info("User is being created");
    }
}
Enter fullscreen mode Exit fullscreen mode

Lombok in Spring Boot Applications

Lombok’s features are particularly useful in Spring Boot applications, where boilerplate code can easily accumulate in services, repositories, and models. Here’s an example of using Lombok in a Spring Boot service:

import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;

@Slf4j
@Service
@RequiredArgsConstructor
public class UserService {

    private final UserRepository userRepository;

    public void saveUser(User user) {
        log.info("Saving user: {}", user);
        userRepository.save(user);
    }
}
Enter fullscreen mode Exit fullscreen mode

In this example:

• @RequiredArgsConstructor generates a constructor for userRepository, which Spring Boot uses for dependency injection.
• @Slf4j adds a logger, so we don’t need to manually define a logging instance.
Enter fullscreen mode Exit fullscreen mode




Conclusion

Lombok is an invaluable tool for Java developers. It significantly reduces boilerplate code, keeps classes clean, and enhances productivity. By using annotation processing to manipulate the AST, Lombok injects methods directly during compilation, ensuring code conciseness without compromising on functionality.

When combined with Spring Boot, Lombok streamlines development even further. Annotations like @Data, @Builder, and @Slf4j provide a powerful way to write clean, maintainable code that’s easy to extend and debug.

If you’re working in Java, Lombok is a must-have in your toolkit. Why write more code than necessary when Lombok can handle it?

Top comments (2)

Collapse
 
khmarbaise profile image
Karl Heinz Marbaise

I recommend to use records instead of Lombok...

Collapse
 
be11amer profile image
Bellamer

Of course, but only for Java 16+ . Some legacy code base and more customisation Lombok is the solution. It’s always nice to get more than one solution I think. I will also write about Records, as it is nativ to java. :)