forked from dotnet/csharplang
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Replace spec content with links (dotnet#5815)
* Replace spec content with links Replace the content of the spec folder with a list of anchored links to the ECMA draft spec. Fixes dotnet/docs#27789 (final task). * proofread * add one clarifying note. * missed one link * Add note for tag I created a tag with the latest commit before merging this PR. Add that note to the text with all links.
- Loading branch information
1 parent
e34e61f
commit d74f8f1
Showing
20 changed files
with
906 additions
and
23,150 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,197 +1,16 @@ | ||
# Arrays | ||
|
||
An array is a data structure that contains a number of variables which are accessed through computed indices. The variables contained in an array, also called the elements of the array, are all of the same type, and this type is called the element type of the array. | ||
|
||
An array has a rank which determines the number of indices associated with each array element. The rank of an array is also referred to as the dimensions of the array. An array with a rank of one is called a ***single-dimensional array***. An array with a rank greater than one is called a ***multi-dimensional array***. Specific sized multi-dimensional arrays are often referred to as two-dimensional arrays, three-dimensional arrays, and so on. | ||
|
||
Each dimension of an array has an associated length which is an integral number greater than or equal to zero. The dimension lengths are not part of the type of the array, but rather are established when an instance of the array type is created at run-time. The length of a dimension determines the valid range of indices for that dimension: For a dimension of length `N`, indices can range from `0` to `N - 1` inclusive. The total number of elements in an array is the product of the lengths of each dimension in the array. If one or more of the dimensions of an array have a length of zero, the array is said to be empty. | ||
|
||
The element type of an array can be any type, including an array type. | ||
|
||
## Array types | ||
|
||
An array type is written as a *non_array_type* followed by one or more *rank_specifier*s: | ||
|
||
```antlr | ||
array_type | ||
: non_array_type rank_specifier+ | ||
; | ||
non_array_type | ||
: type | ||
; | ||
rank_specifier | ||
: '[' dim_separator* ']' | ||
; | ||
dim_separator | ||
: ',' | ||
; | ||
``` | ||
|
||
A *non_array_type* is any *type* that is not itself an *array_type*. | ||
|
||
The rank of an array type is given by the leftmost *rank_specifier* in the *array_type*: A *rank_specifier* indicates that the array is an array with a rank of one plus the number of "`,`" tokens in the *rank_specifier*. | ||
|
||
The element type of an array type is the type that results from deleting the leftmost *rank_specifier*: | ||
|
||
* An array type of the form `T[R]` is an array with rank `R` and a non-array element type `T`. | ||
* An array type of the form `T[R][R1]...[Rn]` is an array with rank `R` and an element type `T[R1]...[Rn]`. | ||
|
||
In effect, the *rank_specifier*s are read from left to right before the final non-array element type. The type `int[][,,][,]` is a single-dimensional array of three-dimensional arrays of two-dimensional arrays of `int`. | ||
|
||
At run-time, a value of an array type can be `null` or a reference to an instance of that array type. | ||
|
||
### The System.Array type | ||
|
||
The type `System.Array` is the abstract base type of all array types. An implicit reference conversion ([Implicit reference conversions](conversions.md#implicit-reference-conversions)) exists from any array type to `System.Array`, and an explicit reference conversion ([Explicit reference conversions](conversions.md#explicit-reference-conversions)) exists from `System.Array` to any array type. Note that `System.Array` is not itself an *array_type*. Rather, it is a *class_type* from which all *array_type*s are derived. | ||
|
||
At run-time, a value of type `System.Array` can be `null` or a reference to an instance of any array type. | ||
|
||
### Arrays and the generic IList interface | ||
|
||
A one-dimensional array `T[]` implements the interface `System.Collections.Generic.IList<T>` (`IList<T>` for short) and its base interfaces. Accordingly, there is an implicit conversion from `T[]` to `IList<T>` and its base interfaces. In addition, if there is an implicit reference conversion from `S` to `T` then `S[]` implements `IList<T>` and there is an implicit reference conversion from `S[]` to `IList<T>` and its base interfaces ([Implicit reference conversions](conversions.md#implicit-reference-conversions)). If there is an explicit reference conversion from `S` to `T` then there is an explicit reference conversion from `S[]` to `IList<T>` and its base interfaces ([Explicit reference conversions](conversions.md#explicit-reference-conversions)). For example: | ||
```csharp | ||
using System.Collections.Generic; | ||
|
||
class Test | ||
{ | ||
static void Main() { | ||
string[] sa = new string[5]; | ||
object[] oa1 = new object[5]; | ||
object[] oa2 = sa; | ||
|
||
IList<string> lst1 = sa; // Ok | ||
IList<string> lst2 = oa1; // Error, cast needed | ||
IList<object> lst3 = sa; // Ok | ||
IList<object> lst4 = oa1; // Ok | ||
IList<string> lst5 = (IList<string>)oa1; // Exception | ||
IList<string> lst6 = (IList<string>)oa2; // Ok | ||
} | ||
} | ||
``` | ||
|
||
The assignment `lst2 = oa1` generates a compile-time error since the conversion from `object[]` to `IList<string>` is an explicit conversion, not implicit. The cast `(IList<string>)oa1` will cause an exception to be thrown at run-time since `oa1` references an `object[]` and not a `string[]`. However the cast `(IList<string>)oa2` will not cause an exception to be thrown since `oa2` references a `string[]`. | ||
|
||
Whenever there is an implicit or explicit reference conversion from `S[]` to `IList<T>`, there is also an explicit reference conversion from `IList<T>` and its base interfaces to `S[]` ([Explicit reference conversions](conversions.md#explicit-reference-conversions)). | ||
|
||
When an array type `S[]` implements `IList<T>`, some of the members of the implemented interface may throw exceptions. The precise behavior of the implementation of the interface is beyond the scope of this specification. | ||
|
||
## Array creation | ||
|
||
Array instances are created by *array_creation_expression*s ([Array creation expressions](expressions.md#array-creation-expressions)) or by field or local variable declarations that include an *array_initializer* ([Array initializers](arrays.md#array-initializers)). | ||
|
||
When an array instance is created, the rank and length of each dimension are established and then remain constant for the entire lifetime of the instance. In other words, it is not possible to change the rank of an existing array instance, nor is it possible to resize its dimensions. | ||
|
||
An array instance is always of an array type. The `System.Array` type is an abstract type that cannot be instantiated. | ||
|
||
Elements of arrays created by *array_creation_expression*s are always initialized to their default value ([Default values](variables.md#default-values)). | ||
|
||
## Array element access | ||
|
||
Array elements are accessed using *element_access* expressions ([Array access](expressions.md#array-access)) of the form `A[I1, I2, ..., In]`, where `A` is an expression of an array type and each `Ix` is an expression of type `int`, `uint`, `long`, `ulong`, or can be implicitly converted to one or more of these types. The result of an array element access is a variable, namely the array element selected by the indices. | ||
|
||
The elements of an array can be enumerated using a `foreach` statement ([The foreach statement](statements.md#the-foreach-statement)). | ||
|
||
## Array members | ||
|
||
Every array type inherits the members declared by the `System.Array` type. | ||
|
||
## Array covariance | ||
|
||
For any two *reference_type*s `A` and `B`, if an implicit reference conversion ([Implicit reference conversions](conversions.md#implicit-reference-conversions)) or explicit reference conversion ([Explicit reference conversions](conversions.md#explicit-reference-conversions)) exists from `A` to `B`, then the same reference conversion also exists from the array type `A[R]` to the array type `B[R]`, where `R` is any given *rank_specifier* (but the same for both array types). This relationship is known as ***array covariance***. Array covariance in particular means that a value of an array type `A[R]` may actually be a reference to an instance of an array type `B[R]`, provided an implicit reference conversion exists from `B` to `A`. | ||
|
||
Because of array covariance, assignments to elements of reference type arrays include a run-time check which ensures that the value being assigned to the array element is actually of a permitted type ([Simple assignment](expressions.md#simple-assignment)). For example: | ||
```csharp | ||
class Test | ||
{ | ||
static void Fill(object[] array, int index, int count, object value) { | ||
for (int i = index; i < index + count; i++) array[i] = value; | ||
} | ||
|
||
static void Main() { | ||
string[] strings = new string[100]; | ||
Fill(strings, 0, 100, "Undefined"); | ||
Fill(strings, 0, 10, null); | ||
Fill(strings, 90, 10, 0); | ||
} | ||
} | ||
``` | ||
|
||
The assignment to `array[i]` in the `Fill` method implicitly includes a run-time check which ensures that the object referenced by `value` is either `null` or an instance that is compatible with the actual element type of `array`. In `Main`, the first two invocations of `Fill` succeed, but the third invocation causes a `System.ArrayTypeMismatchException` to be thrown upon executing the first assignment to `array[i]`. The exception occurs because a boxed `int` cannot be stored in a `string` array. | ||
|
||
Array covariance specifically does not extend to arrays of *value_type*s. For example, no conversion exists that permits an `int[]` to be treated as an `object[]`. | ||
|
||
## Array initializers | ||
|
||
Array initializers may be specified in field declarations ([Fields](classes.md#fields)), local variable declarations ([Local variable declarations](statements.md#local-variable-declarations)), and array creation expressions ([Array creation expressions](expressions.md#array-creation-expressions)): | ||
|
||
```antlr | ||
array_initializer | ||
: '{' variable_initializer_list? '}' | ||
| '{' variable_initializer_list ',' '}' | ||
; | ||
variable_initializer_list | ||
: variable_initializer (',' variable_initializer)* | ||
; | ||
variable_initializer | ||
: expression | ||
| array_initializer | ||
; | ||
``` | ||
|
||
An array initializer consists of a sequence of variable initializers, enclosed by "`{`" and "`}`" tokens and separated by "`,`" tokens. Each variable initializer is an expression or, in the case of a multi-dimensional array, a nested array initializer. | ||
|
||
The context in which an array initializer is used determines the type of the array being initialized. In an array creation expression, the array type immediately precedes the initializer, or is inferred from the expressions in the array initializer. In a field or variable declaration, the array type is the type of the field or variable being declared. When an array initializer is used in a field or variable declaration, such as: | ||
```csharp | ||
int[] a = {0, 2, 4, 6, 8}; | ||
``` | ||
it is simply shorthand for an equivalent array creation expression: | ||
```csharp | ||
int[] a = new int[] {0, 2, 4, 6, 8}; | ||
``` | ||
|
||
For a single-dimensional array, the array initializer must consist of a sequence of expressions that are assignment compatible with the element type of the array. The expressions initialize array elements in increasing order, starting with the element at index zero. The number of expressions in the array initializer determines the length of the array instance being created. For example, the array initializer above creates an `int[]` instance of length 5 and then initializes the instance with the following values: | ||
```csharp | ||
a[0] = 0; a[1] = 2; a[2] = 4; a[3] = 6; a[4] = 8; | ||
``` | ||
|
||
For a multi-dimensional array, the array initializer must have as many levels of nesting as there are dimensions in the array. The outermost nesting level corresponds to the leftmost dimension and the innermost nesting level corresponds to the rightmost dimension. The length of each dimension of the array is determined by the number of elements at the corresponding nesting level in the array initializer. For each nested array initializer, the number of elements must be the same as the other array initializers at the same level. The example: | ||
```csharp | ||
int[,] b = {{0, 1}, {2, 3}, {4, 5}, {6, 7}, {8, 9}}; | ||
``` | ||
creates a two-dimensional array with a length of five for the leftmost dimension and a length of two for the rightmost dimension: | ||
```csharp | ||
int[,] b = new int[5, 2]; | ||
``` | ||
and then initializes the array instance with the following values: | ||
```csharp | ||
b[0, 0] = 0; b[0, 1] = 1; | ||
b[1, 0] = 2; b[1, 1] = 3; | ||
b[2, 0] = 4; b[2, 1] = 5; | ||
b[3, 0] = 6; b[3, 1] = 7; | ||
b[4, 0] = 8; b[4, 1] = 9; | ||
``` | ||
|
||
If a dimension other than the rightmost is given with length zero, the subsequent dimensions are assumed to also have length zero. The example: | ||
```csharp | ||
int[,] c = {}; | ||
``` | ||
creates a two-dimensional array with a length of zero for both the leftmost and the rightmost dimension: | ||
```csharp | ||
int[,] c = new int[0, 0]; | ||
``` | ||
|
||
When an array creation expression includes both explicit dimension lengths and an array initializer, the lengths must be constant expressions and the number of elements at each nesting level must match the corresponding dimension length. Here are some examples: | ||
```csharp | ||
int i = 3; | ||
int[] x = new int[3] {0, 1, 2}; // OK | ||
int[] y = new int[i] {0, 1, 2}; // Error, i not a constant | ||
int[] z = new int[3] {0, 1, 2, 3}; // Error, length/initializer mismatch | ||
``` | ||
|
||
Here, the initializer for `y` results in a compile-time error because the dimension length expression is not a constant, and the initializer for `z` results in a compile-time error because the length and the number of elements in the initializer do not agree. | ||
# Arrays | ||
|
||
This content has moved to the [`dotnet/csharpstandard`](https://github.com/dotnet/csharpstandard) repository. | ||
The list below provides links to each heading in this section. The links specify the C# 6 branch, which is version when the specifications merged. | ||
|
||
> To view the text of the Microsoft spec before merging with the ECMA text, checkout the [ms-spec-text](https://github.com/dotnet/csharplang/releases/tag/ms-spec-text) tag in this repository. | ||
- <a id="arrays"></a>[Arrays](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#16-arrays) | ||
- <a id="array-types"></a>[Array types](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#162-array-types) | ||
- <a id="the-systemarray-type"></a>[The System.Array type](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#1622-the-systemarray-type) | ||
- <a id="arrays-and-the-generic-ilist-interface"></a>[Arrays and the generic IList interface](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#1623-arrays-and-the-generic-collection-interfaces) | ||
- <a id="array-creation"></a>[Array creation](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#163-array-creation) | ||
- <a id="array-element-access"></a>[Array element access](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#164-array-element-access) | ||
- <a id="array-members"></a>[Array members](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#165-array-members) | ||
- <a id="array-covariance"></a>[Array covariance](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#166-array-covariance) | ||
- <a id="array-initializers"></a>[Array initializers](https://github.com/dotnet/csharpstandard/blob/draft-v6/standard/arrays.md#167-array-initializers) |
Oops, something went wrong.