5. Types


As mentioned earlier in the article, C# supports value types and reference types. Value types include simple data type such as int, char, and bool. Reference types include object, class, interface, and delegate.

A value type contains the actual value of the object. That means the actual data is stored in the variable of a value type, whereas a reference type variable contains the reference to the actual data.

Value Types


Value types reference the actual data and declared by using their default constructors. The default constructor of these types returns a zero- initialized instance of the variable. The value types can further be categorized instance of the variable. The value types can further be categorized into many subcategories, described in the following sections.

Simple Types


Simple types include basic data types such as int, char, and bool. These types have a reserved keyword corresponding to one class of a CLS type defined in the System class. For example, the keyword int aliases the System.Int32 type, and the keyword long aliases the System.Int64 type. Table 3 describes simple types. 


Table 3 simple types
 
C# TYPE ALIAS
CLS TYPE
SIZE BITS
SUFFIX
DESCRIPTION
RANGE
sbyte
Sbyte
8
N/a
Singed byte
-128 to 127
byte
Byte
8
N/a
Unsigned byte
0 to 255
short
Int16
16
N/a
Short integer
-32,768 to 32,767
ushort
unit16
16
N/a
Unsigned short integer
0 to 65,535
int
Int32
32
N/a
Integer
-2,147,483,648 to 2,17483,648
uint
uint32
32
U
Unsigned integer
0 to 4,294,967,295
long
Int64
64
L
Long integer
-9223372036854775808 to 9223372036854775808
ulong
uint64
64
N/a
Unsigned long integer
0 to 18,446,744,073,709,551,615
char
char
16
N/a
Unicode character
any valid character, e.g., a,*, \x0058 (hex), or\u0058 (Unicode)
float
single
32
F
Floating point integer

double
double
64
D
Double floating point integer

bool
boolean
1
N/a
Logical true/false value
True/false
decimal
decimal
128
M
Used for financial and monetary calculations





 

One feature of simple types is that you can assign single direct values to these types. Listing 9 shows some assignment examples.

Listing 9. Simple type example

using System;
namespace ToStringSamp
{
class Test
{
static void Main(string[ ] args)
{
int num1 =12;
float num2 =3.05f;
double num3 = 3.5;
bool bl = true;

Console.WriteLine(num1.ToString());
Console.WriteLine(num2.ToString());
Console.WriteLine(num3.ToString());
Console.WriteLine(bl.ToString());
}
}
}

Struct Type


A struct type, or structure type, can declare constructors, constants, fields, methods, properties, indexers, operators, and nested types. Structure types are similar to classes, but they’re lightweight objects with no inheritance mechanism.

However, all structures inherit from the Object class.

In listing 10, your struct CarRec uses a record for a car with three members: name, model, and year.

Listing 10. a struct type example

using System;

struct CarRec
{
public string Name;
public string Model;
public int Year;
}

class TestStructureType
{
public static void Main ()
{
CarRec rec;
rec.Name ="Honda";
rec.Model ="Accord";
rec.Year = 1999;
Console.WriteLine("Car Name: " +rec.Name);
Console.WriteLine("Car Modal: " +rec.Model );
Console.WriteLine("Car: "+rec.Year);
}
}

Figure  5 shows the output of listing 10.
 
Figure 5. Output of listing 10

Enum data types


The enum data types are useful when you need to represent a set of multiple values. A good example of an enumeration is a list of colors:

enum ColorEnum {black, red, green};

Enum types are limited to long, int, short and byte.

This code declares an enum    ColorEnum with members black, red, and green:

//black is 0, red is 1, green is 2.
enum ColorEnum{black, red, green};

You can also set your associated value to an e num type such as:

enum ColorEnum {black =0, red =1, green =2};

By default, enum associated value starts with 0 and increases by 1 for the next defined member. If you assign your value, the default value of the next e num type member will be the value of current member plus 1. For example, in this code the value of green is 7;

enum ColorEnum {black =0, red =6, green };

Reference Types


A reference type is a reference to an instance type. The main reference types are class, array, interface, delegate, and event. A null value is assigned to a reference type by default. A type assigned to a null value means the absence of an instance of that type.

Class Type


A class type defines a data structure that can have members in the form of methods, properties, indexers, events, constructors, operators, and delegates. The class keyword is used to create a class type. You can add methods, properties, indexers, delegates, and events to the class. Listing 11 shows an properties, indexers, delegates, and events to the class. Listing 11 shows an example of a class type.

Listing 11 Class Type example

// Define Class 1
public class class1:Object
{
private void Method1()
{
Console.WriteLine("1 method" );
}
}

The new keyword creates access to the class type. After creating an instance, you can use the dot (.) operator to access its members, as shows here:

Class1 cls1 = new class1();
cls1.Method1();

I’ll return to the discussion of classes later in this article.

Interface Type


An interface type is an abstract base class, which is a skeleton of a class and doesn’t implement the members that it defines. Only the derived class of an interface can implement the members of the interface. Interfaces can contain methods, properties, events, and indexers.

In listing 12 MyInterface is an interface that defines the method TestMethod.MyClass is derived from MyInterface, and you implement the MyMethod method in MyClass.

Listing 12. The interface type example

using System;
interface MyInterface
{
      void TestMethod();
}

class MyClass:MyInterface
{
public static void Main()
{
MyClass cls=new MyClass();
cls.TestMethod();
}
public void TestMethod()
{
Console.WriteLine("Test Method");
}
}

A class can also implement multiple interfaces. Listing 13 defines two interfaces, MyInterface and MyInterface2.MyClass is inherited from these interfaces. You must implement these interfaces in the inherited class. If you don’t implement an interface in the derived class, the complier gives an error message.

For example, if you don’t implement the method test method TestMethod2 of MyInterface2 in Myclass, the compiler returns this message: “Myclass does not implement the interface member ‘MyInterface2. TestMethod2 (int, int)’.“

Listing 13. Multiple interfaces

using System;
interface MyInterface
{
    void TestMethod();
}

interface MyInterface2
{
    int TestMethod2(int a, int b);
}

class MyClass : MyInterface, MyInterface2
{
    public static void main()
    {
        int num1 = 23;
        int num2 = 6;
        MyClass cls = new MyClass();
        cls.TestMethod();
        int tot = cls.TestMethod2(num1, num2);
        Console.WriteLine(tot.ToString());
    }
    public void TestMethod()
    {
        Console.WriteLine("test method");
    }

    public int TestMethod2(int a, int b)
    {
        return a + b;
    }
}

Delegates Types


Delegate types are mainly are used with the class events. A delegate type encapsulates a method with a certain signature, called a callable entity. Delegates are the typesafe and secure version of function pointers (callback functionality).

Delegate instances are not aware of the methods they encapsulate; they’re aware only and return type.

There are three steps in defining and using a delegate:  declaration syntax. For example, this code:

delegate void MyDelegate():


Declares a delegate named MyDelegate that no arguments and returns void.
The next step is to create an instance of delegate and call it:

MyDelegate del =new MyDelegate(TestMethod);
del();

Listing 14 shows an example of delegate.

Listing 14. An example of delegate.

delegate void MyDelegate();
class Test
{
static void TestMethod()
{
System.Console.WriteLine("Test Method called");
}
static void Main()
{
MyDelegate del = new MyDelegate(TestMethod);
del();
}
}

Event Types


The event keyword defines an event. An eventype enables an object or class to provide notification of an event from the system. An instance of a delegate type encapsulates the callable entities. The EventHandler class defines a delegate definition. For example:

public delegate void EventHandler(object sender, System.Event Args e);
public event EventHandler Click;
...........

I’ll discuss events in more detail in the “Class Members” section of this article.

Array Types


An array type is a sequential set of any of the other types. Arrays can be either single- or multidimensional. Both rectangular and jagged arrays are supported a jagged array has elements that don’t necessarily have the same length. A rectangular array is multidimensional, and all of its subarrays have the same length. With arrays, all of the elements must be of the same base type. In C#, the lower index of an array starts with 0, and the upper index is number of item minus 1.

You can initialize array item either during the creation of an array or later by referencing array item, as shown here:

int[] nums = new int[5];
int[0] = 1;
int[1] = 2;
int[2] = 3;
int[3] = 4;
int[4] = 5;

Or here

int[] nums = new int {1,2,3,4,5,};

Listing 15 shows an example of single- dimensional arrays.

Listing 15. Single dimensional array example

class Test
{
      static void Main()
      {
            //array of integers
            int[] nums = new int[5];
            // Array of strings
            string[ ] names = new string[2];

            for(int i =0; i< nums.Length; i++)
                  nums[i] = i+2;
            names[0] = "Mahesh";
            names[1] = "Chand";
for (int i = 0; i< nums.Length; i++)
System.Console.WriteLine ("num[{0}] = {1}", i, nums[i] );
System.Console.WriteLine
(names[0].ToString() + " " + names[1].ToString() );
}
}

The following is an example is an example of multiple, rectangular, and jagged arrays:

char[] arr1 =new char[] {‘a‘, ‘b‘, ‘c’};
int[,] arrr2 = new int[,] {{2,4}, {3, 5}};
//rectangular array declaration
int [, ,]arr3= new int[2,4,6];
// also rectangular
int[][]jarr = new int[3][];
//jagged array declaration
jarr[0] = new int[] {1,2,3};
jarr[1] = new int[] {1,2,3,4,5,6};
jarr[2] = new int[] {1,2,3,4,5,6,7,8,9};  

Sorting Searching, and Copying Arrays


The array class defines functionalities for creating, manipulating, searching, shorting, and copying arrays. Table4 lists and describes some of the array class properties.

Table 4. The array class properties

PROPERTY
 DESRIPITION
Length
Number of items in an array
Rank
Number of dimensions in an array
IsFixedLength
Indicates if an array is of fixed length
IsReadOnly
Indicates if an array is read-only



Table 5 describes some of the array Class methods.

Table 5. The array class methods

METHOD
DESCRIPTION
BinarySearch
Searches for an element using Binary search algorithm
Clear
Removes all elements of an array and set reference to null
Copy
Copies a section of one array to another
CreateInstance
Initializes a new instance of an array
Reverse
Reverses the order of array elements
Sort
Sorts the elements of an array
Clone
Creates a shallow copy of an array
CopyTo
Copies all elements from 1 D array to another
GetLength
Returns number of items in an array
GetValue
Gets a value at a specified location
SetValue
Sets a value at a specified location



The Copy method copies one-array section to another array section. However, this method only works for single-dimensional array. Listing 16 shows a sample of coping array items from one array to another.

Listing 16. Copying array sample

using System;

public class ArraySample
{
      public static void Main()
      {

            // Create and initialize a new arrays
            int[] intArr = new int[5] {1,2,3,4,5};
            Object[] objArr = new Object[5] {10,20,30,40,50};

            foreach (int i in intArr)
            {
                  Console.Write(i);
                  Console.Write(",");
            }
            Console.WriteLine();
            foreach (Object i in objArr )
            {
                  Console.Write (i);
                  Console.Write (",");
            }
            Console.WriteLine();
            // Copy one first 3 elements of intArr to objArr
            Array.Copy(intArr, objArr,3);

            Console.WriteLine("After coping" );
            foreach (int i in intArr)
            {
                  Console.Write(i);
                  Console.Write(" , ");
            }
            Console.WriteLine( );
            foreach (Object i in objArr)
            {
                  Console.Write(i);
                  Console.Write(" ,");
            }
            Console.WriteLine( );
      }
}

The Sort and Reverse methods of the array class are useful when you need to sort and reverse array elements. Listing 17 shows how to sort and reverse arrays.

Listing 17. Reversing and sorting array elements

using System;

public class ArraySample
{
    public static void Main()
    {

        // Create and initialize a new array instance.
        Array strArr = Array.CreateInstance(typeof(string), 3);
        strArr.SetValue("Mahesh", 0);
        strArr.SetValue("chand", 1);
        strArr.SetValue("Test Array", 2);

        // Display the values of the array.
        Console.WriteLine("Initial Array values:");
        for (int i = strArr.GetLowerBound(0);
i <= strArr.GetUpperBound(0); i++)
            Console.WriteLine(strArr.GetValue(i));

        //sort the value of the array.
        Array.Sort(strArr);

        Console.WriteLine("After sorting:");
        for (int i = strArr.GetLowerBound(0);
i <= strArr.GetUpperBound(0); i++)
            Console.WriteLine(strArr.GetValue(i));

        // Reverse values of the array.
        Array.Reverse(strArr);

        for (int i = strArr.GetLowerBound(0); i <= strArr.GetUpperBound(0); i++)
            Console.WriteLine(strArr.GetValue(i));

    }
}

Type Conversions


C# supports two kinds of type conversions: implicit conversions and explicit conversions. Some of the predefined types define predefined conversions, such as converting from an int type to a long type.

Implicit conversions are conversions in which one type can directly and safely are converted to another type. Generally, small range type converts to large range type. As an example, you’ll examine the process of converting from an int type to a long type. In this conversion, there is no loss of data, as shown in Listing 18.

Listing 18. Conversion example

using System;
class ConversionSamp
{
      static void Main()
{
int num1 = 123;
long num2 = num1;
Console.WriteLine(num1.ToString());
Console.WriteLine(num2.ToString());
}
}

Casting performs explicit conversions. There may be a chance of data loss or even some errors in explicit conversions. For example, converting a long value to an integer would result in data loss.

This is an example of an explicit conversion:

long num1 = Int64.MaxValue;
int num2 =(int)num1;
Console.WriteLine(num1.ToString());
Console.WriteLine(num2.ToString());

The process of converting from a value type to a reference type is called boxing. Boxing is an implicit conversion. Listing 19 shows an example of boxing.

Listing 19. Boxing example

using System;
class ConversionSamp
{
    static void Main()
    {
        int num1 = 123;
        Object obj = num1;

        Console.WriteLine(num1.ToString());
        Console.WriteLine(obj.ToString());
    }
}

The process of converting from a reference type to a value type is called unboxing. Listing 20 shows an example of unboxing.

Listing 20. Unboxing example

using System;
class ConversionSamp
{
    static void Main()
    {
        Object obj = 123;
        int num1 = (int)obj;

        Console.WriteLine(num1.ToString());
        Console.WriteLine(obj.ToString());
    }
}






Previous           Next