Skip to content

Bits of .NET

Daily micro-tips for C#, SQL, performance, and scalable backend engineering.

  • Asp.Net Core
  • C#
  • SQL
  • JavaScript
  • CSS
  • About
  • ErcanOPAK.com
  • No Access
  • Privacy Policy
C#

C#: Use Tuples for Multiple Return Values

- 19.03.26 - ErcanOPAK

📦 Return Multiple Values Elegantly

Creating class for one method? Using out parameters? Ugly. Tuples return multiple values cleanly.

The Old Ways (Painful)

// ❌ Out parameters (ugly)
public void GetStats(out int count, out decimal total, out decimal avg)
{
    count = 100;
    total = 5000m;
    avg = 50m;
}

// Ugly usage
GetStats(out int count, out decimal total, out decimal avg);

// ❌ Creating class for single use
public class Stats
{
    public int Count { get; set; }
    public decimal Total { get; set; }
    public decimal Average { get; set; }
}

public Stats GetStats() { /* ... */ }

The Tuple Way (Elegant)

// ✅ Named tuple
public (int Count, decimal Total, decimal Average) GetStats()
{
    var count = 100;
    var total = 5000m;
    var avg = 50m;
    
    return (count, total, avg);
}

// Clean usage
var stats = GetStats();
Console.WriteLine($"Count: {stats.Count}");
Console.WriteLine($"Total: {stats.Total:C}");

// Or deconstruct
var (count, total, avg) = GetStats();
Console.WriteLine($"Average: {avg:C}");

🎯 Real-World Examples

// Parse with validation
public (bool Success, int Value) TryParseInt(string input)
{
    if (int.TryParse(input, out int result))
        return (true, result);
    return (false, 0);
}

var (success, value) = TryParseInt("123");
if (success)
    Console.WriteLine($"Parsed: {value}");

// Min/Max in one pass
public (int Min, int Max) GetMinMax(int[] numbers)
{
    return (numbers.Min(), numbers.Max());
}

var (min, max) = GetMinMax(new[] { 1, 5, 3, 9, 2 });

// Coordinates
public (double Lat, double Lng) GetLocation()
{
    return (37.7749, -122.4194);
}

var location = GetLocation();
Console.WriteLine($"Lat: {location.Lat}, Lng: {location.Lng}");

💡 Pro Tips

  • Always name tuple elements: (int Count, decimal Total) not (int, decimal)
  • Deconstruct selectively: var (count, _) = GetStats(); to ignore values
  • Pattern matching: Works with switch expressions
  • When not to use: If returned from many places, use proper class

“Deleted 15 single-use DTOs. Replaced with named tuples. Code is cleaner, less ceremony, same type safety. Tuples are underrated.”

— Senior C# Developer

Related posts:

Span for High-Performance Parsing

Converting a List to a comma separated string in C#

Hide GridView Column on server-side

Post Views: 6

Post navigation

C#: Use String Interpolation with Format Specifiers
C#: Use Expression-Bodied Members for One-Liners

Leave a Reply Cancel reply

Your email address will not be published. Required fields are marked *

April 2026
M T W T F S S
 12345
6789101112
13141516171819
20212223242526
27282930  
« Mar    

Most Viewed Posts

  • Get the User Name and Domain Name from an Email Address in SQL (950)
  • How to add default value for Entity Framework migrations for DateTime and Bool (858)
  • Get the First and Last Word from a String or Sentence in SQL (836)
  • How to select distinct rows in a datatable in C# (805)
  • How to make theater mode the default for Youtube (751)
  • Add Constraint to SQL Table to ensure email contains @ (578)
  • How to enable, disable and check if Service Broker is enabled on a database in SQL Server (564)
  • Average of all values in a column that are not zero in SQL (531)
  • How to use Map Mode for Vertical Scroll Mode in Visual Studio (489)
  • Find numbers with more than two decimal places in SQL (447)

Recent Posts

  • C#: Use Init-Only Setters for Immutable Objects After Construction
  • C#: Use Expression-Bodied Members for Concise Single-Line Methods
  • C#: Enable Nullable Reference Types to Eliminate Null Reference Exceptions
  • C#: Use Record Types for Immutable Data Objects
  • SQL: Use CTEs for Readable Complex Queries
  • SQL: Use Window Functions for Advanced Analytical Queries
  • .NET Core: Use Background Services for Long-Running Tasks
  • .NET Core: Use Minimal APIs for Lightweight HTTP Services
  • Git: Use Cherry-Pick to Apply Specific Commits Across Branches
  • Git: Use Interactive Rebase to Clean Up Commit History Before Merge

Most Viewed Posts

  • Get the User Name and Domain Name from an Email Address in SQL (950)
  • How to add default value for Entity Framework migrations for DateTime and Bool (858)
  • Get the First and Last Word from a String or Sentence in SQL (836)
  • How to select distinct rows in a datatable in C# (805)
  • How to make theater mode the default for Youtube (751)

Recent Posts

  • C#: Use Init-Only Setters for Immutable Objects After Construction
  • C#: Use Expression-Bodied Members for Concise Single-Line Methods
  • C#: Enable Nullable Reference Types to Eliminate Null Reference Exceptions
  • C#: Use Record Types for Immutable Data Objects
  • SQL: Use CTEs for Readable Complex Queries

Social

  • ErcanOPAK.com
  • GoodReads
  • LetterBoxD
  • Linkedin
  • The Blog
  • Twitter
© 2026 Bits of .NET | Built with Xblog Plus free WordPress theme by wpthemespace.com