8.2. Multi-dimensional Arrays
An array can have any number of dimensions. For example, one-dimension (1D), 2-dimension (2D), 3-dimension (3D) as shown below [1].
Since arrays use indices to access the elements, you must specify the indexing position within the array for data manipulation. In a 1d array you specify just the index, in a 2d array, you specify row and column, and in a 3d array you specify plane, row, and column. You can have higher dimension arrays, but they are just difficult to imagine.
8.2.1. Rectangular Arrays (Two Dimensional)
While a one-dimensional array works for a sequence of data, you need something more for a two dimensional table, where data values vary over both row and column.
If we have a table of integers, for instance with three rows and four columns:
2 4 7 55
3 1 8 10
6 0 49 12
We could declare an array variable of the right size as
int[,] table = new int[3, 4];
Multiple indices are separated by commas inside the square brackets.
In declaring an array type, no indices are included so the [,]
indicates a two dimensional array. Where the new
object is being created,
the values inside the square brackets give each dimension.
In general the notation for a two dimensional array declaration is:
type
[, ]
variableName
and to create a new array with default values:
new type
[
intExpression1,
intExpression2]
where the expressions evaluate to integers for the dimensions.
To assign the 8 in the table above, consider that it is in the second row in normal counting, but we start array indices at 0, so there are rows 0, 1, and 2, and the 8 has row index 1. Again starting with index 0 for columns, the 8 is at index 1. We can assign a value to that position with
table[1, 2] = 8;
In fact a two dimensional array just needs two indices that could mean
anything. For instance, it was just the standard convention,
calling the left index the row. They could have been switched everywhere, and
assume the notation table[column, row]
:
int[,] table = new int[4, 3];
table[2, 1] = 8;
but we will stick with the original [row, column]
model.
Data indexed by more than two integer indices can be stored in a higher dimension array, with more indices between the square braces. We will only consider two dimensional arrays in the examples here.
A shorthand for initializing all the data in the table, analogous to initializing one dimensional arrays is:
int[, ] table = { {2, 4, 7, 55},
{3, 1, 8, 10},
{6, 0, 49, 12} };
All rows must be the same length when using this notation.
Often two dimensional arrays, like one dimensional arrays, are processed in loops. Multiple dimension arrays are often processed in nested loops. We could print out this table using columns 5 spaces wide with the code:
for (int r = 0; r < 3; r++) {
for (int c = 0; c < 4; c++) {
Console.Write("{0, 5}", table[r, c]);
}
Console.WriteLine();
}
If we wanted to make a more general function out of that code, we have a problem: the number of rows and columns were literal values that we knew. We need something more general. For one dimensional arrays we had the Length property, but now there are more than one lengths!
The following csharprepl sequence illustrates the syntax needed:
> int[, ] table = { {2, 4, 7, 55},
> {3, 1, 8, 10},
> {6, 0, 49, 12} };
> table.Length; // get the total of elements
12
> Console.WriteLine(table.GetLength(0)); // get number of elements of the 1st dimension (row) of array
3
> Console.WriteLine(table.GetLength(1)); // get length of the 2nd dimension (column) of array
4
> foreach (int n in table) {
> Console.Write($"{n} ");
> }
2 4 7 55 3 1 8 10 6 0 49 12
>
Note:
The meaning for the Length property, is now the total number of elements, (3)(4) = 12.
The separate method
GetLength
is needed to find the number of rows and columns. The entries in the list of array indices in the multi-dimensional array notation are themselves indexed to provide theGetLength
method parameter for each dimension. In this case index 0 gives the row length (left index of the array notation), and 1 gives the column length (right index of the array notation).The
foreach
statement behavior is consistent with theLength
property, giving all the elements, row by row. More generally, the rightmost indices vary most rapidly as theforeach
statement iterates through all elements.The Console.Write($”{n} “) uses String Interpolation.
Just replacing 3 and 4 by table.GetLength(0)
and table.GetLength(1)
in the
table printing code would make it general.
A more elaborate table might include row and column sums:
2 4 7 55 | 68
3 1 8 10 | 22
6 0 49 12 | 67
---------------------
11 5 64 77 | 157
For example, the following function from example file print_table/print_table.cs, prints out a table of integers neatly, including row and column sums. It illustrates a number of things. It shows the interplay between one and two dimensional arrays, since the row and column sums are just one dimensional arrays.
Now that we are using arrays,
we can easily look at the same collection of data repeatedly.
It is possible to look at the data one time to just see its maximum
width, and then go through again and print data using a column
width that is just large enough for the longest numbers.
When looking through the data for string lengths, the row and column
structure is not important, so the code illustrates foreach
loops
to chug through all the data.
The code refers once to the earlier StringOfReps
in StringOfReps
for the row of dashes setting off the column sums:
static void PrintTableAndSums(int[,] t)
{
int[] rowSum = new int[t.GetLength(0)],
colSum = new int[t.GetLength(1)];
int sum = 0;
for (int r = 0; r < t.GetLength(0); r++) {
for (int c = 0; c < t.GetLength(1); c++) {
rowSum[r] += t[r, c];
colSum[c] += t[r, c];
}
sum += rowSum[r];
}
int width = (""+sum).Length;
foreach (int n in t) {
width = Math.Max(width, (""+n).Length);
}
foreach (int n in colSum) {
width = Math.Max(width, (""+n).Length);
}
foreach (int n in rowSum) {
width = Math.Max(width, (""+n).Length);
}
string format = "{0," + width + "} ";
for (int r = 0; r < t.GetLength(0); r++) {
for (int c = 0; c < t.GetLength(1); c++) {
Console.Write(format, t[r, c]);
}
Console.WriteLine("| " + format, rowSum[r]);
}
string reps = StringOfReps("-",(width+1)*(t.GetLength(1)+1) + 1);
Console.WriteLine(reps);
for (int c = 0; c < t.GetLength(1); c++) {
Console.Write(format, colSum[c]);
}
Console.WriteLine("| " + format, sum);
}
8.2.2. Advanced topic: Array of Arrays
Since you can have an array of any type, it is also possible to have an array of arrays (jagged arrays). The notational difference to a multi-dimensional array is just that each index is enclosed in separate square brackets, with no commas.
For example, you could create a table with shape like the initial example in Rectangular Arrays (Two Dimensional), less the last two elements to show the jagged array:
// declare the array of three elements
int[][] table2 = new int[3][];
// initialize the array elements
table2[0] = new int[4] {2, 3, 7, 55}
table2[1] = new int[4] {3, 1, 8, 10}
table2[2] = new int[2] {6, 0}
// access an element by row and column index
Console.WriteLine(table2[0][3]); // output 55
// display the array elements:
for (int i = 0; i < table2.Length; i++)
{
System.Console.Write($"Element [{i}] Array: ");
for (int j = 0; j < table2[i].Length; j++)
Console.Write($"{table2[i][j]} ");
Console.WriteLine();
} // output:
// Element [0] Array: 2 3 7 55
// Element [1] Array: 3 1 8 10
// Element [2] Array: 6 0
This takes somewhat more memory and is longer to access than the multidimensional arrays. For most uses, when you want to refer to a rectangular table of data, like above, this new version has little to offer.
There are a couple of reasons why you might want this format.
First you may have functions that operate on one dimensional arrays,
and individual rows of table2
are one dimensional arrays:
table2[1]
has type int[]
. On the other hand, with the
earlier table
with type int[,]
, references to part of
the array must always include a comma, so tabel[1]
would
not refer to a row, but would be a syntax error.
Also in an array of arrays, since each row is an independent array, their lengths can vary. Here is code to make a doubly indexed “triangular” collection (of 0 values). Each row must be separately created as a new array:
int[][] tri = new int[4][]; //create four null int[] elements
for (int i = 0; i < tri.Length; i++) { // Length 4 (rows)
tri[i] = new int[i+1]; // each row has a different length
}
With tri
constructed as above:
tri[3, 3]
would given a compiler error: no changing[][]
to[, ]
.tri[3][3]
would refer to an element.tri[1][3]
would given a run-time out-of-bounds error. sincetri[1]
is an array of length 2.
Footnotes