skills/virtuslab/agent-skills/java-best-practices

java-best-practices

SKILL.md

Java Best Practices by VirtusLab

Opinionated, modern Java skill. Targets Java 21+ and explicitly cuts off outdated patterns. When writing or reviewing Java code, follow these guidelines strictly.

Core Philosophy

  • Composed Method Pattern is the single most impactful practice in Java. Every method should do one thing, at one level of abstraction, and be short enough to read in one glance. This pattern alone transforms unreadable code into clean code.
  • Effective Java by Joshua Bloch remains the authoritative reference. Its principles are still current. When in doubt, consult it.
  • Use library code. Do not reimplement what the standard library or a well-established library already provides. This is one of the most violated and most impactful principles.
  • Composition over inheritance. Always. Use interfaces with default methods and delegation instead of deep class hierarchies.
  • GRASP principles (General Responsibility Assignment Software Patterns) guide object-oriented design: Information Expert, Creator, Controller, Low Coupling, High Cohesion, Polymorphism, Pure Fabrication, Indirection, Protected Variations.

Modern Java (21+) — Use These Features

Use modern Java features aggressively. Do not write pre-Java 17 style code.

Records for Data Objects

Use record for all immutable data carriers. Do NOT use Lombok @Value, @Data, @Getter, or hand-written POJOs for data objects.

// WRONG — legacy style
@Value
public class User {
    String username;
    String email;
}

// WRONG — hand-written boilerplate
public class User {
    private final String username;
    private final String email;
    // constructor, getters, equals, hashCode, toString...
}

// CORRECT
public record User(String username, String email) {}

Lombok is acceptable only for @Builder on non-record classes where the builder pattern is genuinely needed. For everything else, use records or plain Java.

Sealed Classes for Domain Modeling (ADTs)

Use sealed classes and interfaces to model algebraic data types. This gives exhaustiveness checking in switch expressions.

public sealed interface PaymentResult
    permits PaymentSuccess, PaymentFailure, PaymentPending {
}

public record PaymentSuccess(String transactionId, Money amount) implements PaymentResult {}
public record PaymentFailure(String reason, ErrorCode code) implements PaymentResult {}
public record PaymentPending(String transactionId) implements PaymentResult {}

Switch Expressions with Pattern Matching

Use switch expressions (not statements) with pattern matching. Prefer exhaustive switches.

// CORRECT
String message = switch (result) {
    case PaymentSuccess success ->
        "Paid %s".formatted(success.amount());
    case PaymentFailure failure ->
        "Failed: %s".formatted(failure.reason());
    case PaymentPending pending ->
        "Pending: %s".formatted(pending.transactionId());
};

// WRONG — old-style switch statement with break
switch (result.getType()) {
    case SUCCESS:
        message = "Paid";
        break;
    // ...
}

Null Handling Inside switch (Java 21+)

// CORRECT — null is an explicit case
return switch (status) {
    case null -> "unknown";
    case ACTIVE -> "active";
    case PAUSED -> "paused";
};

// WRONG — null handled outside the switch
if (status == null) return "unknown";
return switch (status) {
    case ACTIVE -> "active";
    case PAUSED -> "paused";
};

Pattern Matching for instanceof

// CORRECT
if (obj instanceof String s && !s.isBlank()) {
    process(s);
}

// WRONG
if (obj instanceof String) {
    String s = (String) obj;
    process(s);
}

Text Blocks

Use text blocks for multi-line strings (SQL, JSON, HTML, etc.).

String query = """
    SELECT u.id, u.name
    FROM users u
    WHERE u.active = true
    ORDER BY u.name
    """;

var for Local Variables

Use var for local variables when the type is obvious from the right-hand side. Do not use var when it reduces readability.

// CORRECT — type is obvious
var users = userRepository.findAll();
var mapper = new ObjectMapper();

// WRONG — type is not obvious, use explicit type
var result = process(data);

Record Patterns (Java 21)

Destructure records directly in instanceof and switch expressions.

// CORRECT
if (obj instanceof Point(int x, int y)) {
    System.out.println(x + ", " + y);
}

String desc = switch (shape) {
    case Circle(double r) -> "Circle r=" + r;
    case Rect(double w, double h) -> "Rect " + w + "x" + h;
};

// WRONG — old style: cast + field access
if (obj instanceof Point p) {
    System.out.println(p.x() + ", " + p.y());
}

Unnamed Variables (Java 22)

Use _ for unused variables, catch parameters, and lambda parameters.

// CORRECT
try { ... } catch (Exception _) { ... }
try (var _ = ScopedValue.where(KEY, value)) { ... }
list.stream().map(_ -> "constant").toList();

// WRONG
try { ... } catch (Exception ignored) { ... }
try { ... } catch (Exception e) { ... } // when e is never used

Immutable Collections

Use factory methods for immutable collections. Never use Arrays.asList() or new ArrayList<>() for fixed collections.

// CORRECT
var roles = List.of("ADMIN", "USER");
var config = Map.of("timeout", 30, "retries", 3);
var tags = Set.of("java", "backend");

// WRONG
var roles = Arrays.asList("ADMIN", "USER");
var roles = new ArrayList<>(List.of("ADMIN", "USER"));
var roles = Collections.unmodifiableList(Arrays.asList("ADMIN", "USER"));

Sequenced Collections (Java 21)

Use getFirst(), getLast(), reversed() instead of index arithmetic.

// CORRECT
var first = list.getFirst();
var last = list.getLast();
var rev = list.reversed();

// WRONG
var first = list.get(0);
var last = list.get(list.size() - 1);
var rev = new ArrayList<>(list); Collections.reverse(rev);

String Utilities (Java 11+)

Use modern String methods. Do not reimplement what the standard library already provides.

// CORRECT
str.isBlank()               // replaces str.trim().isEmpty()
str.strip()                 // Unicode-aware trim
"ha".repeat(3)              // "hahaha"
"Hello %s".formatted(name) // replaces String.format(...)
str.lines()                 // Stream<String> of lines

// WRONG
str.trim().isEmpty()
String.format("Hello %s", name)

Null Safety

Nulls are not acceptable inside the system. Things are non-null by default. Do not let nulls propagate beyond system boundaries (external APIs, database results, user input).

  • Do NOT annotate everything with @NonNull — that pollutes the code. Instead, treat everything as non-null by default.
  • Use @Nullable only when something genuinely can be null (rare, at system edges).
  • If you receive null from an external source, convert it at the boundary: to an Optional, a default value, or throw early.
  • Never pass null as a method argument. Never return null from a method. Use Optional for methods that may have no result.
// CORRECT — Optional for methods that may return nothing
public Optional<User> findByEmail(String email) {
    return Optional.ofNullable(repository.get(email));
}

// WRONG — returning null
public User findByEmail(String email) {
    return repository.get(email); // might return null
}

Optional

  • Use Optional as return type for public methods that may have no result. It communicates intent: "this method might return nothing, and you must handle it."
  • Never use Optional as a field type, method parameter, or in collections.
  • Never call Optional.get() without checking. Use orElseThrow(), orElse(), map(), flatMap(), ifPresent().
// CORRECT
userService.findByEmail(email)
    .map(User::username)
    .orElseThrow(() -> new UserNotFoundException(email));

// WRONG
User user = userService.findByEmail(email).get();

Error Handling

Use Java's exception mechanism. Do not try to turn Java into Go or C with error return codes or Result wrapper types. Java has checked and unchecked exceptions — use them.

  • Use unchecked exceptions (extending RuntimeException) for programming errors and unexpected failures.
  • Use checked exceptions sparingly, only when the caller genuinely must handle the failure and there is a reasonable recovery path.
  • Never swallow exceptions. Always log or rethrow.
  • Never catch Exception or Throwable generically unless at the top-level error boundary (e.g., controller advice, global handler).
  • Use specific exception types. Create custom domain exceptions when the standard ones do not fit.
  • Use multi-catch to avoid duplicated catch blocks.
// CORRECT
catch (IOException | SQLException e) {
    log.error("Data access failed", e);
}

// WRONG — duplicated handlers
catch (IOException e) { log.error("...", e); }
catch (SQLException e) { log.error("...", e); }
// CORRECT
public User getUser(UserId id) {
    return userRepository.findById(id)
        .orElseThrow(() -> new UserNotFoundException(id));
}

// WRONG — returning null on error
public User getUser(UserId id) {
    try {
        return userRepository.findById(id).orElse(null);
    } catch (Exception e) {
        return null;
    }
}

Naming Conventions

  • Classes: PascalCase — nouns (UserService, PaymentProcessor).
  • Methods: camelCase — verbs (findByEmail, processPayment, calculateTotal).
  • Variables: camelCase — nouns (userId, orderItems).
  • Constants: UPPER_SNAKE_CASE (MAX_RETRY_COUNT, DEFAULT_TIMEOUT).
  • Packages: lowercase, dot-separated (com.company.project.domain.user).
  • Booleans: prefix with is, has, should, can (isValid(), hasPermission(), shouldRetry()).
  • Collections: plural names (users, orders, products).
  • No abbreviations in public APIs. userRepository not userRepo. transactionManager not txMgr.
  • Methods should not have surprising side effects. The name must accurately describe what the method does.

Code Style & Formatting

  • Use one consistent style across the project: either standard Java Coding Conventions or IntelliJ defaults. The specific choice does not matter — consistency does.
  • Keep classes package-private by default. Only expose what is needed as public.
  • Apply final to variables when immutability is intended.
  • Use method references over lambdas when possible: User::userId instead of u -> u.userId().
  • Organize imports: standard library, third-party, internal packages. No wildcard imports.
  • If the project includes Spotless, run the formatter after every code change (only for edited files). For Gradle: ./gradlew spotlessApply. For Maven: mvn spotless:apply.

Streams & Collections

Use Java Streams for collection transformations. Prefer filter(), map(), flatMap(), toList().

// CORRECT
var activeEmails = users.stream()
    .filter(User::isActive)
    .map(User::email)
    .toList();

// WRONG — manual loop for simple transformation
var activeEmails = new ArrayList<String>();
for (User user : users) {
    if (user.isActive()) {
        activeEmails.add(user.getEmail());
    }
}

Never perform side effects in intermediate operations (like map, filter, flatMap) — these should be pure functions. Terminal forEach is the right place for side effects. Prefer Iterable.forEach() over stream().forEach() when no intermediate operations are needed.

// WRONG — side effect in intermediate operation
users.stream()
    .map(user -> { emailService.send(user); return user.email(); })
    .toList();

// WRONG — unnecessary stream() when there is no pipeline
users.stream().forEach(user -> emailService.send(user));

// CORRECT — Iterable.forEach for simple side effects
users.forEach(user -> emailService.send(user));

// CORRECT — pure filter, side effect only in terminal forEach
users.stream()
    .filter(User::isActive)
    .forEach(user -> emailService.send(user));

// CORRECT — plain loop when you need break/continue or checked exceptions
for (var user : users) {
    emailService.send(user);
}

Testing

Consistent Test Naming

Pick one naming convention for the entire project and enforce it. Recommended format:

methodName_shouldExpectedBehavior_whenCondition

Examples:

@Test
void findByEmail_shouldReturnUser_whenUserExists() { ... }

@Test
void findByEmail_shouldThrowNotFoundException_whenUserDoesNotExist() { ... }

@Test
void processPayment_shouldReturnSuccess_whenBalanceSufficient() { ... }

Do NOT mix camelCase, snake_case, given/when/then, and should styles in the same project. Consistency matters more than the specific convention.

Test Structure

  • Use JUnit 5 as the default testing framework.
  • Spock Framework (Groovy-based) is an acceptable alternative if the team prefers BDD-style specifications. It is powerful but requires Groovy proficiency.
  • Follow Arrange-Act-Assert (or Given-When-Then in Spock).
  • Each test should test one behavior. No multi-assert tests covering unrelated behaviors.
  • Use @DisplayName sparingly — a well-named test method is self-documenting.

TDD

Test-Driven Development produces better APIs. Writing tests first forces you to design the public interface before the implementation. Even if not doing strict TDD, write tests immediately after writing the method signature and before the implementation body.

Test Independence

Tests must be independent and repeatable. No shared mutable state between tests. No ordering dependencies.

Concurrency

Virtual Threads (Project Loom)

Use Virtual Threads for I/O-bound concurrent work. They eliminate the need for reactive frameworks for most use cases.

// CORRECT — Virtual Threads
try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
    var future = executor.submit(() -> fetchData(url));
    return future.get();
}

// WRONG — reactive chains for simple I/O
Mono.fromCallable(() -> fetchData(url))
    .subscribeOn(Schedulers.boundedElastic())
    .flatMap(data -> process(data))
    .subscribe();

Immutability Enables Safe Concurrency

Writing proper OOP with small, immutable objects in small methods with limited scope is the best concurrency strategy. Small objects are created and garbage-collected quickly, reducing contention.

Structured Concurrency (Java 25)

Use StructuredTaskScope to manage lifetimes of forked subtasks as a unit. Ensures that subtasks are reliably cancelled and joined when the scope closes.

// CORRECT
try (var scope = new StructuredTaskScope.ShutdownOnFailure()) {
    var user = scope.fork(() -> fetchUser(id));
    var order = scope.fork(() -> fetchOrder(id));
    scope.join().throwIfFailed();
    return new Response(user.get(), order.get());
}

// WRONG — manual executor management, no automatic lifecycle
var executor = Executors.newFixedThreadPool(2);
var f1 = executor.submit(() -> fetchUser(id));
var f2 = executor.submit(() -> fetchOrder(id));
executor.shutdown();
return new Response(f1.get(), f2.get());

What NOT to Use

  • Do NOT use RxJava. It is overly complex, hard to debug, and unnecessary with Virtual Threads. If you are in a project that uses RxJava, do not introduce more of it.
  • Avoid CompletableFuture chains when Virtual Threads can express the same logic sequentially.
  • Do not prematurely introduce reactive patterns. Use them only when you have proven back-pressure requirements.

Modern I/O

Use Files convenience methods and Path.of() factory. Avoid BufferedReader/Writer boilerplate for simple operations.

// CORRECT
String content = Files.readString(Path.of("config.json"));
Files.writeString(Path.of("output.txt"), data);
var path = Path.of("src", "main", "java");

// WRONG
var br = new BufferedReader(new FileReader("config.json"));
// ... manual read loop ...
var path = Paths.get("src", "main", "java");

Use the built-in HttpClient for HTTP calls. Do not add HttpURLConnection boilerplate.

// CORRECT
var client = HttpClient.newHttpClient();
var request = HttpRequest.newBuilder(URI.create(url)).GET().build();
var response = client.send(request, HttpResponse.BodyHandlers.ofString());

// WRONG — HttpURLConnection with manual stream handling
var conn = (HttpURLConnection) new URL(url).openConnection();
conn.setRequestMethod("GET");
var reader = new BufferedReader(new InputStreamReader(conn.getInputStream()));
// ...

Date and Time

Always use java.time (Java 8+). Never use Date, Calendar, or SimpleDateFormat.

// CORRECT
LocalDate today = LocalDate.now();
LocalDate date = LocalDate.of(2025, Month.JANUARY, 15);
Instant now = Instant.now();
long days = ChronoUnit.DAYS.between(start, end);
Duration duration = Duration.ofMinutes(30);

// WRONG
Date date = new Date();
Calendar cal = Calendar.getInstance();
cal.set(2025, 0, 15); // zero-indexed months

Logging

  • Use SLF4J (org.slf4j.Logger). Never use System.out.println() or System.err.println().
  • Use parameterized logging — never string concatenation.
// CORRECT
log.info("Processing order {} for user {}", orderId, userId);

// WRONG
log.info("Processing order " + orderId + " for user " + userId);
  • Log levels:
    • error — exceptions and failures requiring attention.
    • warn — recoverable issues, degraded behavior.
    • info — significant business events (order placed, payment processed).
    • debug — detailed technical information for troubleshooting.
    • trace — very fine-grained, typically disabled in production.

Project Structure

  • Separate domain code from infrastructure code in the package structure.
  • Keep configuration classes in dedicated config packages.
  • Follow consistent package naming: com.company.project.<domain>.<layer> (e.g., com.acme.shop.order.repository).
  • One public class per file. No multi-class files except for tightly coupled inner classes.

JVM & GC

  • Do not tune GC parameters unless you have measured a problem. The default GC (G1GC in modern JVMs) is good enough for the vast majority of applications.
  • Writing proper OOP with small, short-lived objects is better than any GC tuning.
  • Profile before optimizing. Use JFR (Java Flight Recorder) and JMC (Java Mission Control) for performance analysis.

Deprecated Patterns — Do NOT Use

These patterns are outdated and must not be used in new code:

Deprecated Pattern Modern Replacement
Lombok @Value, @Data, @Getter, @Setter for data classes record
Arrays.asList() List.of()
new ArrayList<>(Arrays.asList(...)) List.of() or new ArrayList<>(List.of(...))
Collections.unmodifiableList() List.copyOf() or List.of()
RxJava / Project Reactor for simple I/O Virtual Threads
Old switch statements with break Switch expressions
Manual instanceof + cast Pattern matching instanceof
String concatenation with + for multi-line Text blocks
Anonymous inner classes for functional interfaces Lambdas / method references
synchronized blocks for simple cases java.util.concurrent locks or Virtual Threads
Returning null from methods Optional or throw exception
System.out.println() SLF4J logging
Raw types (List instead of List<String>) Always use generics
Date, Calendar java.time API (LocalDate, Instant, etc.)
StringBuffer StringBuilder (or text blocks / formatted())
str.trim().isEmpty() str.isBlank()
String.format(...) "...".formatted(...)
list.get(list.size() - 1) list.getLast()
Collections.reverse(new ArrayList<>(list)) list.reversed()
catch (Exception ignored) catch (Exception _)
Duplicate catch blocks Multi-catch catch (A | B e)
BufferedReader + loop for file reading Files.readString(Path.of(...))
Paths.get(...) Path.of(...)
HttpURLConnection HttpClient (Java 11)
Manual executor lifecycle for structured tasks StructuredTaskScope (Java 25)
Weekly Installs
22
GitHub Stars
4
First Seen
Jan 27, 2026
Installed on
claude-code22
cursor22
opencode22
gemini-cli8
github-copilot8
cline8