Oggetti (Guida per programmatori C#)
Una definizione di classe o struttura è simile a un progetto iniziale in cui vengono specificate le funzionalità del tipo. Un oggetto è essenzialmente un blocco di memoria che è stato allocato e configurato in base al progetto iniziale. Un programma può creare molti oggetti della stessa classe. Gli oggetti, definiti anche istanze, possono essere archiviati in una variabile denominata o in una matrice o raccolta. Il codice client è il codice che utilizza queste variabili per chiamare i metodi e accedere alle proprietà pubbliche dell'oggetto. In un linguaggio orientato a oggetti quale C#, un tipico programma è costituito da più oggetti che interagiscono dinamicamente.
Nota
I tipi statici si comportano in modo diverso da quanto viene descritto in questo punto.Per ulteriori informazioni, vedere Classi statiche e membri di classi statiche (Guida per programmatori C#).
Istanze della struttura VS. istanze della classe
Poiché le classi sono tipi di riferimento, una variabile di un oggetto classe contiene un riferimento all'indirizzo dell'oggetto sull'heap gestito. Se al primo oggetto viene assegnato un secondo oggetto dello stesso tipo, entrambe le variabili fanno riferimento all'oggetto in corrispondenza di tale indirizzo. Questo punto viene illustrato più dettagliatamente in seguito in questo argomento.
Le istanze di classi vengono create utilizzando l'operatore new. Nel seguente esempio, Person è il tipo e person1 e person 2 sono istanze, o oggetti, di tale tipo.
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
public Person(string name, int age)
{
Name = name;
Age = age;
}
//Other properties, methods, events...
}
class Program
{
static void Main()
{
Person person1 = new Person("Leopold", 6);
Console.WriteLine("person1 Name = {0} Age = {1}", person1.Name, person1.Age);
// Declare new person, assign person1 to it.
Person person2 = person1;
//Change the name of person2, and person1 also changes.
person2.Name = "Molly";
person2.Age = 16;
Console.WriteLine("person2 Name = {0} Age = {1}", person2.Name, person2.Age);
Console.WriteLine("person1 Name = {0} Age = {1}", person1.Name, person1.Age);
// Keep the console open in debug mode.
Console.WriteLine("Press any key to exit.");
Console.ReadKey();
}
}
/*
Output:
person1 Name = Leopold Age = 6
person2 Name = Molly Age = 16
person1 Name = Molly Age = 16
*/
Poiché le strutture sono tipi di valore, una variabile di un oggetto struttura contiene una copia dell'intero oggetto. Anche le istanze di strutture possono essere create tramite l'operatore new, ma non è obbligatorio, come illustrato nell'esempio seguente:
public struct Person
{
public string Name;
public int Age;
public Person(string name, int age)
{
Name = name;
Age = age;
}
}
public class Application
{
static void Main()
{
// Create struct instance and initialize by using "new".
// Memory is allocated on thread stack.
Person p1 = new Person("Alex", 9);
Console.WriteLine("p1 Name = {0} Age = {1}", p1.Name, p1.Age);
// Create new struct object. Note that struct can be initialized
// without using "new".
Person p2 = p1;
// Assign values to p2 members.
p2.Name = "Spencer";
p2.Age = 7;
Console.WriteLine("p2 Name = {0} Age = {1}", p2.Name, p2.Age);
// p1 values remain unchanged because p2 is copy.
Console.WriteLine("p1 Name = {0} Age = {1}", p1.Name, p1.Age);
// Keep the console open in debug mode.
Console.WriteLine("Press any key to exit.");
Console.ReadKey();
}
}
/*
Output:
p1 Name = Alex Age = 9
p2 Name = Spencer Age = 7
p1 Name = Alex Age = 9
*/
La memoria per entrambi gli oggetti p1 sia p2 viene allocata sullo stack di thread. Tale memoria viene recuperata insieme al tipo o al metodo in cui è stata dichiarata. Questo è il motivo per cui le strutture vengono copiate per assegnazione. Per contrasto, la memoria allocata per l'istanza di una classe viene recuperata automaticamente (tramite Garbage Collection) da Common Language Runtime quando tutti i riferimenti all'oggetto sono usciti dall'ambito. Non è possibile eliminare in modo deterministico un oggetto di classe come invece avviene in C++. Per ulteriori informazioni sul sistema di Garbage Collection in .NET Framework, vedere Garbage Collection.
Nota
L'allocazione e la deallocazione di memoria sull'heap gestito sono estremamente ottimizzate in Common Language Runtime.Nella maggior parte dei casi non esistono differenze significative in termini di impatto sulle prestazioni tra l'allocazione di un'istanza di classe sull'heap e l'allocazione di un'istanza di struttura sullo stack.
Identità dell'oggetto VS. uguaglianza di valori
Quando si confrontano due oggetti per verificarne l'uguaglianza, è necessario innanzitutto distinguere se si desidera determinare se le due variabili rappresentano lo stesso oggetto in memoria oppure se i valori di uno o più campi sono equivalenti. Se si intende confrontare valori, è necessario considerare se gli oggetti sono istanze di tipi di valore (strutture) o di tipi di riferimento (classi, delegati, matrici).
Per determinare se due istanze di classe fanno riferimento alla stessa posizione in memoria (ovvero hanno la stessa identità), utilizzare il metodo statico Equals. Object è la classe di base implicita per tutti i tipi valore e i tipi riferimento, incluse strutture e classi definite dall'utente.
Per determinare se i campi di istanza in due istanze di strutture hanno gli stessi valori, utilizzare il metodo ValueType.Equals. Poiché tutte le strutture ereditano implicitamente da ValueType, il metodo viene chiamato direttamente sull'oggetto, come illustrato nell'esempio seguente:
// Person is defined in the previous example.
//public struct Person
//{
// public string Name;
// public int Age;
// public Person(string name, int age)
// {
// Name = name;
// Age = age;
// }
//}
Person p1 = new Person("Wallace", 75);
Person p2;
p2.Name = "Wallace";
p2.Age = 75;
if (p2.Equals(p1))
Console.WriteLine("p2 and p1 have the same values.");
// Output: p2 and p1 have the same values.
L'implementazione ValueType di Equals utilizza la reflection perché deve essere in grado di determinare i tipi di campi presenti in tutte le strutture. Quando si creano strutture, eseguire l'override del metodo Equals per fornire un algoritmo di uguaglianza efficiente specifico del tipo.
- Per determinare se i valori dei campi in due istanze della classe sono uguali, è possibile utilizzare il metodo Equals o l'operatore ==. Tuttavia, utilizzarli solo se la classe ha eseguito il loro override o overload per fornire una definizione personalizzata di cosa significa "uguaglianza" per gli oggetti di quel tipo. È possibile anche che la classe implementi l'interfaccia IEquatable dell'interfaccia IEqualityComparer. Entrambe le interfacce forniscono metodi che possono essere utilizzati per verificare l'uguaglianza dei valori. Durante la progettazione di classi personalizzate che eseguono l'override di Equals, assicurarsi di seguire le linee guida stabilite in Procedura: definire l'uguaglianza di valori per un tipo (Guida per programmatori C#) e in Object.Equals(Object).
Sezioni correlate
Per ulteriori informazioni:
Vedere anche
Riferimenti
Ereditarietà (Guida per programmatori C#)
Operatore new (Riferimenti per C#)