Sdílet prostřednictvím


Anonymní typy

Anonymní typy poskytují pohodlný způsob zapouzdření sady vlastností jen pro čtení do jednoho objektu, aniž byste museli explicitně definovat typ jako první. Název typu je generován kompilátorem a není k dispozici na úrovni zdrojového kódu. Typ každé vlastnosti je odvozen kompilátorem.

Anonymní typy vytvoříte pomocí operátoru new společně s inicializátorem objektů. Další informace o inicializátorech objektů naleznete v tématu Inicializátory objektů a kolekcí.

Následující příklad ukazuje anonymní typ, který je inicializován se dvěma vlastnostmi pojmenovanými Amount a Message.

var v = new { Amount = 108, Message = "Hello" };

// Rest the mouse pointer over v.Amount and v.Message in the following
// statement to verify that their inferred types are int and string.
Console.WriteLine(v.Amount + v.Message);

Anonymní typy se obvykle používají v select klauzuli výrazu dotazu k vrácení podmnožinu vlastností z každého objektu ve zdrojové sekvenci. Další informace odotazch

Anonymní typy obsahují jednu nebo více veřejných vlastností jen pro čtení. Nejsou platné žádné jiné druhy členů třídy, jako jsou metody nebo události. Výraz použitý k inicializaci vlastnosti nemůže být null, anonymní funkce nebo typ ukazatele.

Nejběžnějším scénářem je inicializace anonymního typu s vlastnostmi z jiného typu. V následujícím příkladu předpokládejme, že třída existuje s názvem Product. Třída Product obsahuje Color a Price vlastnosti společně s dalšími vlastnostmi, které vás nezajímají:

class Product
{
    public string? Color {get;set;}
    public  decimal Price {get;set;}
    public string? Name {get;set;}
    public string? Category {get;set;}
    public string? Size {get;set;}
}

Deklarace anonymního typu začíná klíčovým slovem new . Deklarace inicializuje nový typ, který používá pouze dvě vlastnosti z Product. Použití anonymních typů způsobí vrácení menšího množství dat v dotazu.

Pokud nezadáte názvy členů v anonymním typu, kompilátor dává členům anonymního typu stejný název jako vlastnost, která se používá k jejich inicializaci. Zadáte název vlastnosti, která se inicializuje pomocí výrazu, jak je znázorněno v předchozím příkladu. V následujícím příkladu jsou názvy vlastností anonymního typu Color a Price. Instance jsou položky z products kolekce Product typů:

var productQuery =
    from prod in products
    select new { prod.Color, prod.Price };

foreach (var v in productQuery)
{
    Console.WriteLine("Color={0}, Price={1}", v.Color, v.Price);
}

Tip

Pravidlo stylu .NET můžete použít IDE0037 k vynucení, jestli jsou upřednostňované odvozené nebo explicitní názvy členů.

Je také možné definovat pole podle objektu jiného typu: třída, struktura nebo dokonce jiný anonymní typ. Provádí se pomocí proměnné držící tento objekt stejně jako v následujícím příkladu, kde jsou vytvořeny dva anonymní typy pomocí již vytvořených uživatelem definovaných typů. V obou případech bude product pole v anonymním typu shipment a shipmentWithBonus bude obsahovat Product výchozí hodnoty každého pole. bonus Pole bude anonymního typu vytvořeného kompilátorem.

var product = new Product();
var bonus = new { note = "You won!" };
var shipment = new { address = "Nowhere St.", product };
var shipmentWithBonus = new { address = "Somewhere St.", product, bonus };

Při inicializaci proměnné pomocí anonymního typu obvykle deklarujete proměnnou jako implicitně zatypovanou místní proměnnou pomocí proměnné var. Název typu nelze zadat v deklaraci proměnné, protože k základnímu názvu anonymního typu má přístup pouze kompilátor. Další informaceochch var

Pole anonymně zadaných elementů můžete vytvořit kombinací implicitně zadané místní proměnné a implicitně zadaného pole, jak je znázorněno v následujícím příkladu.

var anonArray = new[] { new { name = "apple", diam = 4 }, new { name = "grape", diam = 1 }};

Anonymní typy jsou class typy, které jsou odvozeny přímo z object, a které nelze přetypovat na žádný typ s výjimkou object. Kompilátor poskytuje název pro každý anonymní typ, i když k němu aplikace nemá přístup. Z pohledu modulu CLR (Common Language Runtime) se anonymní typ neliší od jiného typu odkazu.

Pokud dva nebo více anonymních inicializátorů objektů v sestavení určují posloupnost vlastností, které jsou ve stejném pořadí a mají stejné názvy a typy, kompilátor zachází s objekty jako s instancemi stejného typu. Sdílejí stejné informace o typu generovaném kompilátorem.

Anonymní typy podporují nedestruktivní mutování ve formě výrazů. To umožňuje vytvořit novou instanci anonymního typu, kde jedna nebo více vlastností má nové hodnoty:

var apple = new { Item = "apples", Price = 1.35 };
var onSale = apple with { Price = 0.79 };
Console.WriteLine(apple);
Console.WriteLine(onSale);

Nelze deklarovat pole, vlastnost, událost nebo návratový typ metody jako anonymní typ. Podobně nelze deklarovat formální parametr metody, vlastnosti, konstruktoru nebo indexeru jako anonymní typ. Chcete-li předat anonymní typ nebo kolekci, která obsahuje anonymní typy, jako argument metody, můžete deklarovat parametr jako typ object. Použití object pro anonymní typy však porazí účel silného psaní. Pokud je nutné uložit výsledky dotazu nebo je předat mimo hranice metody, zvažte použití obyčejné pojmenované struktury nebo třídy namísto anonymního typu.

Vzhledem k tomu, že Equals anonymní typy a GetHashCode metody jsou definovány z hlediska Equals a GetHashCode metod vlastností, jsou dvě instance stejného anonymního typu stejné, pouze pokud jsou všechny jejich vlastnosti stejné.

Poznámka:

Úroveň přístupnosti anonymního typu je internal, proto dva anonymní typy definované v různých sestaveních nejsou stejného typu. Proto instance anonymních typů nemohou být navzájem rovny, pokud jsou definovány v různých sestaveních, i když mají všechny jejich vlastnosti stejné.

Anonymní typy přepíší metodu ToString , zřetězení názvu a ToString výstupu každé vlastnosti obklopené složenými závorkami.

var v = new { Title = "Hello", Age = 24 };

Console.WriteLine(v.ToString()); // "{ Title = Hello, Age = 24 }"