Why not just use a Tuple or dynamic to return multiple values

Why go through the trouble of implementing a ‘Result Pattern’?

DotNet Full Stack Dev
3 min readMar 23, 2025

In C#, we often need to return more than just raw data — we need to return status, data, errors, and maybe even metadata. You might wonder:

Let’s explore:

  • What the Result Pattern is
  • How it compares to Tuple and dynamic
  • Real-world use cases
  • And why the Result Pattern wins in long-term, clean architecture

🔍 What Is the Result Pattern?

The Result Pattern is a way to return a structured response from a method, typically wrapping:

  • Data (if any)
  • Success/failure status
  • Optional error or message info

📌Explore more at: https://dotnet-fullstack-dev.blogspot.com/
🌟 Clapping would be appreciated! 🚀

👇 A Typical Result<T> Pattern:

1️⃣ Using Tuple:

public class Result<T>
{
public bool IsSuccess { get; }
public string Error { get; }
public T Value { get; }

protected Result(bool isSuccess, T value, string error)
{
IsSuccess = isSuccess;
Value = value;
Error = error;
}

public static Result<T> Success(T value) => new(true, value, null);
public static Result<T> Failure(string error) => new(false, default, error);
}

🟢 Pros:

  • Concise
  • Lightweight

🔴 Cons:

  • No encapsulation
  • Easy to misuse (e.g., forget to check IsSuccess)
  • No behavior (logic) can be added
  • Less readable in large projects

2️⃣ Using dynamic:

public dynamic GetUser(int id)
{
var user = db.Users.Find(id);
if (user == null)
return new { IsSuccess = false, Error = "User not found", Data = (User)null };

return new { IsSuccess = true, Error = "", Data = user };
}

🟢 Pros:

  • Super flexible
  • No need to define classes

🔴 Cons:

  • No IntelliSense
  • Prone to runtime errors
  • Hard to refactor
  • Can get messy in team environments

🔥 Why the Result Pattern is Better (Especially in Real-World Scenarios)

✅ 1. Enforces Consistency

Every service or handler returns a standard structure. No surprises.

var result = userService.GetUser(id);
if (!result.IsSuccess)
return BadRequest(result.Error);

return Ok(result.Value);

✅ 2. Encourages Defensive Coding

You must check the result status before proceeding — making bugs less likely.

✅ 3. Easily Extendable

Add a ValidationErrors, HttpStatus, TraceId, or CorrelationId property when needed—without breaking existing contracts.

✅ 4. Testable and Mockable

You can easily mock responses in unit tests:

mockService.Setup(s => s.GetUser(It.IsAny<int>()))
.Returns(Result<User>.Success(new User()));

✅ 5. Works Seamlessly with MediatR / CQRS

The Result pattern shines in clean architectures, especially when using MediatR, Minimal APIs, or Domain-Driven Design.

🤔 Can Tuple or Dynamic Work Like Result Pattern?

Yes — technically, they can.
But should you?

Tuple and dynamic may be okay for:

  • Tiny, internal utilities
  • LINQ expressions
  • Quick prototypes

But for production code, multi-layered apps, and team collaboration?

🚫 They lack:

  • Validation
  • Documentation
  • Refactorability
  • Maintainability

✅ Quick Comparison

💡 Conclusion

Using Tuple or dynamic might seem convenient at first, but as your codebase grows, so does the need for clarity, consistency, and safety.

That’s where the Result Pattern becomes a superpower 💪 — helping you build cleaner, testable, and more maintainable .NET applications.

  • ✅ Use Result<T> for structured, predictable return flows.
  • ⚠️ Use Tuple for quick, internal logic.
  • 🚫 Avoid dynamic unless absolutely necessary.

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

--

--

DotNet Full Stack Dev
DotNet Full Stack Dev

Written by DotNet Full Stack Dev

Join me to master .NET Full Stack Development & boost your skills by 1% daily with insights, examples, and techniques! https://dotnet-fullstack-dev.blogspot.com

Responses (1)

Write a response