DekGenius.com
[ Team LiB ] Previous Section Next Section

Recipe 16.3 Navigating Arrays

Problem

You need to iterate through the elements of a single-dimensional, multidimensional, or jagged array using a pointer to that array.

Solution

To enable iteration, we create an unsafe pointer that points to an array. The manipulation of the array can then be performed through this pointer.

To create a pointer to a single-dimension array, declare and initialize the array:

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

and then set a pointer, arrayPtr, to the address of the first element in this array (we must use the fixed keyword to pin the array in the managed heap so that the garbage collector does not move it):

fixed(int* arrayPtr = &intArray[0])

Note that this line could also be written as:

fixed(int* arrayPtr = intArray)

without any address of (&) operator or indexer. This is because the array variable always points to the first element, similar to how C++ array pointers operate.

The following code creates and initializes a pointer to a single-dimension array and then displays the last item in that array:

unsafe
{
    int[] intArray = new int[5] {1, 2, 3, 4, 5};
    fixed(int* arrayPtr = &intArray[0])
    {
        Console.WriteLine(*(arrayPtr + 4)); //Display the last value '5'
    }
}

Creating a pointer to an array of enumeration values is very similar:

unsafe
{
    Colors[] intArray = new Colors[2] {Colors.Red, Colors.Blue};
    fixed(Colors* arrayPtr = &intArray[0])
    {
        // Use arrayPtr here
    }
}

where Colors is declared as follows:

public enum Colors{Red, Green, Blue}

The last element of the array can then be displayed with the following code:

Console.WriteLine(*(arrayPtr + intArray.GetLength(0) - 1));

Creating a pointer to a multidimensional array is performed by declaring and initializing a multidimensional array:

int[,] intMultiArray = new int[2,5] {{1,2,3,4,5},{6,7,8,9,10}};

and then setting a pointer to the address of the first element in this array:

fixed(int* arrayPtr = &intMultiArray[0,0])

For example, the following code creates and initializes a pointer to a multidimensional array and then displays the last item in that array:

unsafe
{
    int[,] intMultiArray = new int[2,5] {{1,2,3,4,5},{6,7,8,9,10}};
    fixed(int* arrayPtr = &intMultiArray[0,0])
    {
        Console.WriteLine(*(arrayPtr + 9)); //Display the last value '10'
    }
}

A jagged array can be pointed to as well, but it is much harder to navigate this type of array using a pointer. This code creates and initializes a pointer to a jagged array and then displays each item in that array:

unsafe
{ 
    int[][] intJaggedArray = new int[3][];
    intJaggedArray[0] = new int[2] {100,200};
    intJaggedArray[1] = new int[3] {300,400,500};
    intJaggedArray[2] = new int[4] {600,700,800,900};
    fixed(int* arrayPtr = &intJaggedArray[0][0])
    {
        for(int counter = -3; counter <= 15; counter++)
        {
            Console.WriteLine(*(arrayPtr + counter)); 
        }
    }
}

This code creates and initializes a pointer to a jagged array whose second array is defined as a multidimensional array, and then displays each item in that array:

unsafe
{ 
    int[][,] intJaggedArray2 = new int[3][,];
    intJaggedArray2[0] = new int[2,1] {{100},{200}};
    intJaggedArray2[1] = new int[3,1] {{300},{400},{500}};
    intJaggedArray2[2] = new int[4,1] {{600},{700},{800},{900}};
    fixed(int* arrayPtr = &intJaggedArray2[0][0,0])
    {
        for(int counter = -5; counter <= 23; counter++)
        {
            Console.WriteLine(*(arrayPtr + counter)); 
        }
    }
}

See Also

See the "Multidimensional Arrays" and "Jagged Arrays" topics and the "Unsafe at the Limit" article in the MSDN documentation.

    [ Team LiB ] Previous Section Next Section