In my last article "Limitations of ArrayLists in C#", I focused on issues of type safety and reusability when using ArrayLists.
In this article, we shall focus on how these issues of type safety and reusability are very nicely handled by Generics. All with the help of code again
- using System;
- using System.Collections.Generic;
- using System.Text;
- namespace GenericsSample
- {
- class Person
- {
- int _Age;
- public int Age
- {
- get { return _Age; }
- set { _Age = value; }
- }
- String _Name;
- public String Name
- {
- get { return _Name; }
- set { _Name = value; }
- }
- String _Address;
- public String Address
- {
- get { return _Address; }
- set { _Address = value; }
- }
- String _Company;
- public String Company
- {
- get { return _Company; }
- set { _Company = value; }
- }
- public Person() { }
- public Person(String Name)
- {
- this.Name = Name;
- this.Age = 0;
- this.Address = String.Empty;
- this.Company = String.Empty;
- }
- public Person(String Name, int Age, String Address)
- {
- this.Name = Name;
- this.Age = Age;
- this.Address = Address;
- }
- }
- class Program
- {
- static void Main(string[] args)
- {
-
-
-
- List<Person> myPerson = new List<Person>();
- myPerson.Add(new Person("Saurabh"));
- myPerson.Add(new Person("Manu"));
- myPerson.Add(new Person("SomeOne", 24, "Gurgaon"));
- myPerson.Add(new Person("SomeoneElse", 24, "Gurgaon"));
-
- foreach (Person p in myPerson)
- {
- Console.WriteLine(p.Name);
- Console.WriteLine(p.Age);
- Console.WriteLine(p.Address);
- Console.WriteLine(p.Company);
- }
- Console.ReadLine();
- }
- }
- }
Two classes can be seen in this code. Class Person is the class of which we want to create list for, and class Program is the main class where we actually create the list of persons and operate upon them.
How Generics tackle the issues posed by ArrayLists?
In the above code example Generic List class has been used to "contain" objects of type Person. At any time we can we can have the Generic List contain any other type, as below:-
-
- List<int> myInts = new List<int>();
- myInts.Add(5);
- myInts.Add(10);
- myInts.Add(20);
- foreach (int x in myInts)
- {
- Console.WriteLine(x);
- }
- Console.ReadLine();
The above code snippet indicates that the same List class can be used to contain any datatype at any point of time, without requiring any kind of extra effort from the programmer's side.
The syntax for using any kind of Generic Class is as under
- GenericClass<T> objT = new GenericClass<T>();
Where T is the datatype that' want to list, and GenericClass is the Generic Class which will wrap our desired datatype (that's the reason, "contains", above has been marked in the double quotes and is marked bold). This Generic Class can be our own custom Generic Class or the ones provided by the .Net Framework.
So technically, T gets replaced by the datatype at compile type. And that's the reasons why a compile time error occurs when casting is not done properly, it will be an InvalidCast Exception while using ArrayLists. Thus Generics enforce type checking at compile time only, making life less difficult.
Performance is another area where Generics make it sweet when compared to ArrayLists. Since T is "replaced" by our datatype at compile time only so, no time and resources are wasted in boxing and unboxing the objects.
Thus Generics are a very powerful and nice feature provided with .Net 2.0. Generics types are found sprinkled throughout the .Net2.0 BCLs; however System.Collections.Generics namespace is chock full of them.