BanksySan
BanksySan

Reputation: 28500

Name ValueTuple properties when creating with new

I know I can name parameters when I create a tuple implicitly like:

var me = (age: 21, favoriteFood: "Custard");

Is it possible to name parameters when a tuple is created explicitly? i.e.

var me = new ValueTuple<int, string>(21, "Custard");

Upvotes: 33

Views: 23121

Answers (5)

Jpsh
Jpsh

Reputation: 1726

... C# 7.0 doesn’t enable the use of custom item names when using the explicit System.ValueTuple<…> data type. Therefore, if you replace var in Example 8 of Figure 1, you’ll end up with warnings that each item name will be ignored.

Source

I'm posting this answer because the closest the docs come to explaining this is:

Data members of ValueTuple types are fields. Data members of Tuple types are properties.

Upvotes: 0

Duikmeester
Duikmeester

Reputation: 1

Was looking for this when using ValueTuple in Linq. I did a F2-rename variable and it created the following syntax:

var query = from x in myList
select (First: x.something, Second: x.other) into g
select new Whatever(g.First, g.Second);

Upvotes: 0

Peter Duniho
Peter Duniho

Reputation: 70652

No, you can't. The ValueTuple types are actually independent of the named field support in C#. The latter works more like named properties for anonymous types. That is, the compiler analyzes the code and generates aliases to the appropriate members according to your declarations and usages. It is through the assignment that the compiler learns the names of the fields. Since the basic constructor syntax doesn't provide a mechanism to name the fields, you can't use that to directly generate a tuple with named fields.

Of course, there are ways you can re-interpret the value returned from the constructor syntax, to assign names to that returned value. I'm assuming you're aware of that approach and are looking for something more direct.

As an example of what I mean by "re-interpret", you could do something like this:

static (int value, string text) ConvertToNamed((int, string) t) => t;

then this would name the fields, in a new variable:

var t1 = new ValueTuple<int, string>(21, "hello");
var t2 = ConvertToNamed(t1);

The variable t1 is stuck with Item1 and Item2. But the compiler will implicitly generate the desired names for the variable t2.

Maybe a better example is one where you don't require the additional method:

(int value, string text) t = new ValueTuple<int, string>(21, "hello");

Again, you're not really naming the fields in the constructor syntax, but they are reinterpreted by the local variable declaration.

This is probably not a serious limitation. In a scenario where there's a desire to have a persistent, easily-assigned name, it's probably better to declare a user-defined type than to use the tuple syntax anyway. You can write deconstructors for user-defined types as well, and declaring types like that means the names are first-class citizens when it comes to reflection, dynamic, etc.

Upvotes: 41

user3875682
user3875682

Reputation: 31

You can cast to named tuple during assignment:

var me = ((int value, string text)) new ValueTuple<int, string>(21, "Custard");

Upvotes: 2

Ian
Ian

Reputation: 3

I don't believe so. Here are docs that I found on ValueTuple:

https://learn.microsoft.com/en-us/dotnet/api/system.valuetuple-7?view=netframework-4.7

I personally have not used the ValueTuble type. I have used the Tuple class before like this:

var tuple = new Tuple<int, string>(21, "Custard");
var number = tuple.Item1;
var st = tuple.Item2;

However I find using Tuples, especially when passed thru methods to be clunky. Always having to know what is in Item1 and Item2 etc. According to the ValueTuple docs, it is used in the same way.

Shouldn't you just make an entity class?

I have used the Anonymous Type which I think will suit your needs.

var anyom = new
{
    age = 21,
    favoriteFood = "Custard"
};

number = anyom.age;
st = anyom.favoriteFood;

Upvotes: -4

Related Questions