2 <clause number="8.2.3" title="Array types" informative="true">
3 <paragraph>Arrays may be single-dimensional or multi-dimensional. Both "rectangular" and "jagged" arrays are supported. </paragraph>
4 <paragraph>Single-dimensional arrays are the most common type. The example <code_example><![CDATA[
9 int[] arr = new int[5];
10 for (int i = 0; i < arr.Length; i++)
12 for (int i = 0; i < arr.Length; i++)
13 Console.WriteLine("arr[{0}] = {1}", i, arr[i]);
16 ]]></code_example>creates a single-dimensional array of <keyword>int</keyword> values, initializes the array elements, and then prints each of them out. The output produced is: <code_example><![CDATA[
22 ]]></code_example></paragraph>
23 <paragraph>The type int[] used in the previous example is an array type. Array types are written using a <non_terminal where="19.1">non-array-type</non_terminal> followed by one or more rank specifiers. The example <code_example><![CDATA[
27 int[] a1; // single-dimensional array of int
28 int[,] a2; // 2-dimensional array of int
29 int[,,] a3; // 3-dimensional array of int
30 int[][] j2; // "jagged" array: array of (array of int)
31 int[][][] j3; // array of (array of (array of int))
34 ]]></code_example>shows a variety of local variable declarations that use array types with <keyword>int</keyword> as the element type. </paragraph>
35 <paragraph>Array types are reference types, and so the declaration of an array variable merely sets aside space for the reference to the array. Array instances are actually created via array initializers and array creation expressions. The example <code_example><![CDATA[
39 int[] a1 = new int[] {1, 2, 3};
40 int[,] a2 = new int[,] {{1, 2, 3}, {4, 5, 6}};
41 int[,,] a3 = new int[10, 20, 30];
42 int[][] j2 = new int[3][];
43 j2[0] = new int[] {1, 2, 3};
44 j2[1] = new int[] {1, 2, 3, 4, 5, 6};
45 j2[2] = new int[] {1, 2, 3, 4, 5, 6, 7, 8, 9};
48 ]]></code_example>shows a variety of array creation expressions. The variables a1, a2 and a3 denote rectangular arrays, and the variable j2 denotes a jagged array. It should be no surprise that these terms are based on the shapes of the arrays. Rectangular arrays always have a rectangular shape. Given the length of each dimension of the array, its rectangular shape is clear. For example, the lengths of a3's three dimensions are 10, 20, and 30, respectively, and it is easy to see that this array contains 10*20*30 elements. </paragraph>
49 <paragraph>In contrast, the variable j2 denotes a "jagged" array, or an "array of arrays". Specifically, j2 denotes an array of an array of <keyword>int</keyword>, or a single-dimensional array of type int[]. Each of these int[] variables can be initialized individually, and this allows the array to take on a jagged shape. The example gives each of the int[] arrays a different length. Specifically, the length of j2[0] is 3, the length of j2[1] is 6, and the length of j2[2] is 9. </paragraph>
51 <note>[Note: In C++, an array declared as <keyword>int</keyword> x[3][5][7] would be considered a three dimensional rectangular array, while in C#, the declaration int[][][] declares a jagged array type. end note]</note>
53 <paragraph>The element type and shape of an array-including whether it is jagged or rectangular, and the number of dimensions it has-are part of its type. On the other hand, the size of the array-as represented by the length of each of its dimensions-is not part of an array's type. This split is made clear in the language syntax, as the length of each dimension is specified in the array creation expression rather than in the array type. For instance the declaration <code_example><![CDATA[
54 int[,,] a3 = new int[10, 20, 30];
55 ]]></code_example>has an array type of int[,,] and an array creation expression of new int[10, 20, 30]. </paragraph>
56 <paragraph>For local variable and field declarations, a shorthand form is permitted so that it is not necessary to re-state the array type. For instance, the example <code_example><![CDATA[
57 int[] a1 = new int[] {1, 2, 3};
58 ]]></code_example>can be shortened to <code_example><![CDATA[
60 ]]></code_example>without any change in program semantics. </paragraph>
61 <paragraph>The context in which an array initializer such as {1, 2, 3} is used determines the type of the array being initialized. The example <code_example><![CDATA[
65 short[] a = {1, 2, 3};
70 ]]></code_example>shows that the same array initializer syntax can be used for several different array types. Because context is required to determine the type of an array initializer, it is not possible to use an array initializer in an expression context without explicitly stating the type of the array. </paragraph>