Unlocking the Full Potential of Optional in Java: Advanced Use Cases for Digital Wallets

Lucas Copque
3 min readFeb 28, 2025

Introduction

Many developers see Optional<T> in Java as a simple way to avoid null values, but its true power goes far beyond that. When used effectively, Optional can help you build cleaner, safer, and more expressive codeโ€”especially in domains like digital wallets, where encapsulation and transaction security are crucial.

This article explores advanced use cases where Optional becomes a game-changer in a digital wallet system, demonstrating how to encapsulate complex business rules efficiently.

๐Ÿ“Œ Note: The examples in this article are based on a fictitious digital wallet system, designed purely for practical learning and demonstrating the power of Optional in Java.

1๏ธโƒฃ Secure Payments with Fraud Prevention

๐Ÿ“Œ Problem: Some transactions may be flagged as suspicious and require additional checks before processing.

๐Ÿ“Œ Solution: The payment process encapsulates rules such as sufficient balance, transaction limits, and fraud detection.

import java.math.BigDecimal;
import java.util.Optional;

public class DigitalWallet {
private BigDecimal balance;
private final BigDecimal transactionLimit = new BigDecimal("5000.00");

public DigitalWallet(BigDecimal initialBalance) {
this.balance = initialBalance;
}

public Optional<Boolean> processPayment(BigDecimal amount, boolean isSuspicious) {
return Optional.of(this)
.filter(w -> !isSuspicious) // Block suspicious transactions
.filter(w -> amount.compareTo(transactionLimit) <= 0) // Respect transaction limits
.flatMap(w -> w.debit(amount)); // Deduct amount if balance is sufficient
}

private Optional<Boolean> debit(BigDecimal amount) {
return Optional.of(this)
.filter(w -> w.balance.compareTo(amount) >= 0)
.map(w -> {
w.balance = w.balance.subtract(amount);
return true;
});
}

public BigDecimal getBalance() {
return balance;
}
}

๐Ÿ”น Benefit: The entire payment process is encapsulated within the wallet, preventing business logic from leaking into the client code.

2๏ธโƒฃ Secure Peer-to-Peer Transfers with Cashback

๐Ÿ“Œ Problem: Transfers between users should ensure sufficient balance, apply cashback rules, and log the transaction.

๐Ÿ“Œ Solution: The entire workflow is handled within DigitalWallet, using map() for a fluent pipeline.

import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;

public class DigitalWallet {
private BigDecimal balance;
private final List<String> transactionHistory = new ArrayList<>();

public DigitalWallet(BigDecimal initialBalance) {
this.balance = initialBalance;
}

public Optional<Boolean> transferTo(DigitalWallet recipient, BigDecimal amount, BigDecimal cashbackPercentage) {
return this.debit(amount)
.map(success -> {
recipient.credit(amount);
recipient.applyCashback(amount, cashbackPercentage);
this.transactionHistory.add("Transfer of $" + amount + " to " + recipient);
return true;
});
}

private Optional<Boolean> debit(BigDecimal amount) {
return Optional.of(this)
.filter(w -> w.balance.compareTo(amount) >= 0)
.map(w -> {
w.balance = w.balance.subtract(amount);
return true;
});
}

private void credit(BigDecimal amount) {
this.balance = this.balance.add(amount);
}

private void applyCashback(BigDecimal amount, BigDecimal cashbackPercentage) {
BigDecimal cashback = amount.multiply(cashbackPercentage).divide(new BigDecimal("100"));
this.credit(cashback);
}

public BigDecimal getBalance() {
return balance;
}
}

๐Ÿ”น Benefit: The entire transaction, including cashback calculation, is encapsulated in a single method.

3๏ธโƒฃ Automatic Subscription Payments with Retry Mechanism

๐Ÿ“Œ Problem: Digital wallets need to automatically pay subscriptions but allow partial payments when funds are low.

๐Ÿ“Œ Solution: Optional enables automatic retries with a 50% fallback payment.

import java.math.BigDecimal;
import java.util.Optional;

public class DigitalWallet {
private BigDecimal balance;

public DigitalWallet(BigDecimal initialBalance) {
this.balance = initialBalance;
}

public Optional<Boolean> paySubscription(BigDecimal monthlyFee) {
return debit(monthlyFee)
.or(() -> retryPayment(monthlyFee));
}

private Optional<Boolean> retryPayment(BigDecimal amount) {
return Optional.of(this)
.filter(w -> w.balance.compareTo(amount.divide(new BigDecimal("2"))) >= 0)
.map(w -> {
w.balance = w.balance.subtract(amount.divide(new BigDecimal("2")));
return true;
});
}

private Optional<Boolean> debit(BigDecimal amount) {
return Optional.of(this)
.filter(w -> w.balance.compareTo(amount) >= 0)
.map(w -> {
w.balance = w.balance.subtract(amount);
return true;
});
}

public BigDecimal getBalance() {
return balance;
}
}

๐Ÿ”น Benefit: The retry mechanism is seamlessly built-in, avoiding manual logic in the client code.

๐Ÿš€ Conclusion: Optional at its Full Power!

By leveraging Optional in Java:

โœ… You eliminate null checks and nested if-else structures.

โœ… You encapsulate business rules within objects, reducing external logic.

โœ… You create expressive and functional pipelines.

โœ… You enable automatic retries and recovery mechanisms.

Using Optional effectively transforms digital wallet management into a secure, reusable, and maintainable system. ๐Ÿš€๐Ÿ’ณ

Sign up to discover human stories that deepen your understanding of the world.

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

Lucas Copque
Lucas Copque

Written by Lucas Copque

Head of Software Engineering at Mobiauto, technology enthusiast with specialization in Java, Spring Boot and Angular.

No responses yet

Write a response