ECMA-334 C# Language Specification17.8: Indexers |
An indexer is a member that enables an object to be indexed in the same way as an array. Indexers are declared using indexer-declarations:
attributes
opt indexer-modifiers
opt indexer-declarator
{
accessor-declarations
}
indexer-modifier
indexer-modifiers
indexer-modifier
type
this [
formal-parameter-list
]
type
interface-type
.
this [
formal-parameter-list
]
An indexer-declaration
may include a set of attributes (24) and a valid combination of the four access modifiers (17.2.3), the new (17.2.2), virtual (17.5.3), override (17.5.4), sealed (17.5.5), abstract (17.5.6), and extern (17.5.7) modifiers.
Indexer declarations are subject to the same rules as method declarations (17.5) with regard to valid combinations of modifiers, with the one exception being that the static modifier is not permitted on an indexer declaration.
The modifiers virtual, override, and abstract are mutually exclusive except in one case. The abstract and override modifiers may be used together so that an abstract indexer can override a virtual one.
The type of an indexer declaration specifies the element type of the indexer introduced by the declaration. Unless the indexer is an explicit interface member implementation, the type is followed by the keyword this. For an explicit interface member implementation, the type is followed by an interface-type
, a ".", and the keyword this. Unlike other members, indexers do not have user-defined names.
The formal-parameter-list
specifies the parameters of the indexer. The formal parameter list of an indexer corresponds to that of a method (17.5.1), except that at least one parameter must be specified, and that the ref and out parameter modifiers are not permitted.
The type of an indexer and each of the types referenced in the formal-parameter-list
must be at least as accessible as the indexer itself (10.5.4).
The accessor-declarations
(17.6.2), which must be enclosed in "{" and "}" tokens, declare the accessors of the indexer. The accessors specify the executable statements associated with reading and writing indexer elements.
Even though the syntax for accessing an indexer element is the same as that for an array element, an indexer element is not classified as a variable. Thus, it is not possible to pass an indexer element as a ref or out argument.
The formal-parameter-list
of an indexer defines the signature (10.6) of the indexer. Specifically, the signature of an indexer consists of the number and types of its formal parameters. The element type and names of the formal parameters are not part of an indexer's signature.
The signature of an indexer must differ from the signatures of all other indexers declared in the same class.
Indexers and properties are very similar in concept, but differ in the following ways:
simple-name
(14.5.2) or a member-access
(14.5.4), whereas an indexer element is accessed through an element-access
(14.5.6.2). Aside from these differences, all rules defined in 17.6.2 and 17.6.3 apply to indexer accessors as well as to property accessors.
When an indexer declaration includes an extern modifier, the indexer is said to be an external indexer. Because an external indexer declaration provides no actual implementation, each of its accessor-declarations
consists of a semicolon.
using System;
class BitArray
{
int[] bits;
int length;
public BitArray(int length) {
if (length < 0) throw new ArgumentException();
bits = new int[((length - 1) >> 5) + 1];
this.length = length;
}
public int Length {
get { return length; }
}
public bool this[int index] {
get {
if (index < 0 || index >= length) {
throw new IndexOutOfRangeException();
}
return (bits[index >> 5] & 1 << index) != 0;
}
set {
if (index < 0 || index >= length) {
throw new IndexOutOfRangeException();
}
if (value) {
bits[index >> 5] |= 1 << index;
}
else {
bits[index >> 5] &= ~(1 << index);
}
}
}
}
class CountPrimes
{
static int Count(int max) {
BitArray flags = new BitArray(max + 1);
int count = 1;
for (int i = 2; i <= max; i++) {
if (!flags[i]) {
for (int j = i * 2; j <= max; j += i) flags[j] = true;
count++;
}
}
return count;
}
static void Main(string[] args) {
int max = int.Parse(args[0]);
int count = Count(max);
Console.WriteLine("Found {0} primes between 1 and {1}", count,
max);
}
}
end example]
using System;
class Grid
{
const int NumRows = 26;
const int NumCols = 10;
int[,] cells = new int[NumRows, NumCols];
public int this[char c, int colm]
{
get {
c = Char.ToUpper(c);
if (c < 'A' || c > 'Z') {
throw new ArgumentException();
}
if (colm < 0 || colm >= NumCols) {
throw new IndexOutOfRangeException();
}
return cells[c - 'A', colm];
}
set {
c = Char.ToUpper(c);
if (c < 'A' || c > 'Z') {
throw new ArgumentException();
}
if (colm < 0 || colm >= NumCols) {
throw new IndexOutOfRangeException();
}
cells[c - 'A', colm] = value;
}
}
}
In This Section: