Real world use cases for C# indexers?
Asked Answered
R

14

49

I've seen lot of examples for c# Indexers, but in what way will it help me in real life situations.

I know the C# guru wouldn't have added this if it wasn't a serious feature, but i cant think of a real world situation (not the foo bar stuff) to use indexers.

Note:I realize a related question exists, but it doesn't help me much.

Respirator answered 2/2, 2010 at 15:15 Comment(7)
Other than in the List-type, the LinkedList-type, the Dictionary-type or the other collections? ;) Often when you have a list of something it is considered useful to provide an indexer, so one can do myList[k] instead of myList.Get(k) or the VB-version myList.Item(k)Jeth
@Jeth Please explain, I'm obviously missing somethingRespirator
So i have a class which contains a Field "col" which is a collection, and when i need to access a particular value in "col", that would be when i will use a IndexerRespirator
See also #1765443Genetics
Yes if your class mimics a storage of some sort or if it contains individual pieces of data you may want to retrieve. String has an indexer, so you can do str[1] to retrieve the second character. FontStorage[key] may make sense, Font[key] probably wouldn't. How the data is stored is usually not important, but the interface (the appearence/representation) determines if an indexer makes sense. That said, I use them very seldom, but they come in handy sometimes. I made a ASP.NET Cache wrapper where Cache["key"] made sense, a real world scenario ;)Jeth
don't forget DataSet and DataTable and DataColumn and DataRow. these all use indexers too!Cabbala
@Jeth you could've posted this as an answer. +1 for sure.Respirator
D
34

The way I look at indexers is that (rightly or wrongly!), accessing something by index should be more efficient than accessing it any other way, because in some way, shape or form, the class whose indexer I'm using stores some form of index that allows it to quickly lookup values when accessed that way.

The classic example is an array, when you access element n of an array by using the code myarray[3], the compiler/interpreter knows how big (memory-wise) elements of the array are and can treat it as an offset from the start of the array. You could also "for(int i = 0; i < myarray.length; i++) { if (i = 3) then { .. do stuff } }" (not that you'd ever want to!), which would be less efficient. It also shows how an array is a bad example.

Say you had a collection class that stores, umm, DVDs, so:

public class DVDCollection
{
    private Dictionary<string, DVD> store = null;
    private Dictionary<ProductId, string> dvdsByProductId = null;

    public DVDCollection()
    {
        // gets DVD data from somewhere and stores it *by* TITLE in "store"
        // stores a lookup set of DVD ProductId's and names in "dvdsByProductid"
        store = new Dictionary<string, DVD>();
        dvdsByProductId = new Dictionary<ProductId, string>();
    }

    // Get the DVD concerned, using an index, by product Id
    public DVD this[ProductId index]  
    {
       var title = dvdsByProductId[index];
       return store[title];
    }
}

Just my 2p, but, like I said,.. I've always considered an "indexer" as being an expedient way of getting data out of something.

Durning answered 2/2, 2010 at 15:28 Comment(0)
S
25

The most obvious examples, as mentioned by Skurmedel, are List<T> and Dictionary<TKey, TValue>. What would you prefer over:

List<string> list = new List<string> { "a", "b", "c" };
string value = list[1]; // This is using an indexer

Dictionary<string, string> dictionary = new Dictionary<string, string>
{
    { "foo", "bar" },
    { "x", "y" }
};
string value = dictionary["x"]; // This is using an indexer

? Now it may be relatively rare for you need to write an indexer (usually when you're creating a collection-like class), but I suspect you use them reasonably frequently.

Sleight answered 2/2, 2010 at 15:26 Comment(2)
Does this mean multiple indexers cannot exist. or in other words can they be overloaded?Respirator
@Vivek Bernard: They can be overloaded.Remonstrance
G
11

Microsoft has an example using an indexer to treat a file as an array of bytes.

public byte this[long index]
{
    // Read one byte at offset index and return it.
    get 
    {
        byte[] buffer = new byte[1];
        stream.Seek(index, SeekOrigin.Begin);
        stream.Read(buffer, 0, 1);
        return buffer[0];
    }
    // Write one byte at offset index and return it.
    set 
    {
        byte[] buffer = new byte[1] {value};
        stream.Seek(index, SeekOrigin.Begin);
        stream.Write(buffer, 0, 1);
    }
}
Genetics answered 2/2, 2010 at 15:21 Comment(1)
I think this is a good example, because indexers allow to retrieve or store items, thereby simulating array behaviour. In this case it is used to persist (serialize and de-serialize) data, hiding the complexity to access a file. It could be a database behind as well. I have also seen that the .NET JSON serializer uses it to simulate JavaScript style arrays allowing you to use strings and numbers as indices in the same class (by overloading the indexer). In the end it is of course syntactical sugar, but it helps to simplify thinks and make them better readable (and more comfortable to use).Larceny
S
7

Let's say you have a collection of objects that you want to be able to index by something other than the order in which it was placed in a collection. In the example below, you can see how you could use the 'Location' property of some object and using the indexer, return all objects in the collection that match you location, or in the second example, all objects that contain a certain Count() of objects.

class MyCollection {

  public IEnumerable<MyObject> this[string indexer] {
    get{ return this.Where(p => p.Location == indexer); }
  }

  public IEnumerable<MyObject> this[int size] {
    get{ return this.Where(p => p.Count() == size);}
  }
}
Smashed answered 2/2, 2010 at 15:19 Comment(0)
L
5

Once .NET got generics, the biggest reason for me to implement an indexer (to implement a strongly-typed collection) went away.

Location answered 2/2, 2010 at 15:19 Comment(1)
Why would the need for an indexer lessen with generics? Or do you mean that you no longer had a need to write your own collections?Jeth
I
3

It is just syntactical sugar for collection type classes. I never had a reason to write such a class. So I think there are very rare uses of it in "real life", because the classes using it are already implemented.

Ides answered 2/2, 2010 at 15:23 Comment(5)
Don’t forget that one mans rarity is another’s requirement. While they are syntacical sugar... you could say that about most of the framework. (Enum, Properties, Operator Overloading, Linq, WCF, etc.)Apprehensible
@Matthew: is there any difference between having an indexer or having a GetElement method? The indexer is shorter to use.Ides
There is no difference between using a Property and a GetProperty method either. If you at the compiled code this happens for you in the back ground. Every keyword we use (beyond the OpCodes for the CPU or Runtime) is form of syntactical sugar... and I, for one, love sugar. For example... when you use a indexer in C# you declare it on the class with the this keyword. On VB.Net you just pick a property and add an input parameter.Apprehensible
@Matthew: Did I say that it is not useful or that I don't like it? I just said that there is nothing so special about it, just to make the usage of collection type classes nicer. I don't know what you want to tell me.Ides
I didn't mean to say either. Your answer just seemed like the many I see about "unneeded" and "unwanted" features. (Along the lines of people that are complaining because LINQ syntax was added.) If you are all for indexers then we agree :) I say use them wherever you need or want to use them. This stuff is all just tools to help make our lives as developers easier.Apprehensible
R
3

In ASP.Net, there are a few different times where an indexer is used such as reading something from any of the Request, Session or Application objects. I've seen often where something is stored in either the Session or Application object only to be used again and again.

Resonant answered 2/2, 2010 at 15:33 Comment(0)
O
2
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace IndexerSample
{
    class FailSoftArray 
    {
        int[] a; // reference to underlying array
        public int Length; // Length is public
        public bool ErrFlag; // indicates outcome of last operation
        // Construct array given its size.
        public FailSoftArray(int size)
        {
            a = new int[size];
            Length = size;
        }
        // This is the indexer for FailSoftArray.
        public int this[int index] 
        {
        // This is the get accessor.
            get
            {
                if (ok(index))
                {
                    ErrFlag = false;
                    return a[index];
                }
                else
                {
                    ErrFlag = true;
                    return 0;
                }
            }
            // This is the set accessor.
            set
            {
                if (ok(index))
                {
                    a[index] = value;
                    ErrFlag = false;
                }
                else ErrFlag = true;
            }
        }
        // Return true if index is within bounds.
        private bool ok(int index)
        {
            if (index >= 0 & index < Length) return true;
            return false;
        }
    }
    class Program
    {
        static void Main(string[] args)
        {
            FailSoftArray fs = new FailSoftArray(5);
            int x;
            // Show quiet failures.
            Console.WriteLine("Fail quietly.");
            for (int i = 0; i < (fs.Length * 2); i++)
                fs[i] = i * 10;
            for (int i = 0; i < (fs.Length * 2); i++)
            {
                x = fs[i];
                if (x != -1) Console.Write(x + " ");
            }
            Console.WriteLine();
            // Now, display failures.
            Console.WriteLine("\nFail with error reports.");
            for (int i = 0; i < (fs.Length * 2); i++)
            {
                fs[i] = i * 10;
                if (fs.ErrFlag)
                    Console.WriteLine("fs[" + i + "] out-of-bounds");
            }
            for (int i = 0; i < (fs.Length * 2); i++)
            {
                x = fs[i];
                if (!fs.ErrFlag) Console.Write(x + " ");
                else
                    Console.WriteLine("fs[" + i + "] out-of-bounds");
            }
            Console.ReadLine();
        }
    }
}
Objective answered 3/5, 2011 at 11:51 Comment(0)
L
2

http://code-kings.blogspot.in/2012/09/indexers-in-c-5.html

Indexers are elements in a C# program that allow a Class to behave as an Array. You would be able to use the entire class as an array. In this array you can store any type of variables. The variables are stored at a separate location but addressed by the class name itself. Creating indexers for Integers, Strings, Boolean etc. would be a feasible idea. These indexers would effectively act on objects of the class.

Lets suppose you have created a class indexer that stores the roll number of a student in a class. Further, lets suppose that you have created an object of this class named obj1. When you say obj1[0], you are referring to the first student on roll. Likewise obj1[1] refers to the 2nd student on roll.

Therefore the object takes indexed values to refer to the Integer variable that is privately or publicly stored in the class. Suppose you did not have this facility then you would probably refer in this way(which would be longer):

obj1.RollNumberVariable[0]
obj1.RollNumberVariable[1]. 

where RollNumberVariable would be the Integer variable that refers to the Roll Number of the current student object.

For more details http://code-kings.blogspot.in/2012/09/indexers-in-c-5.html

Landsturm answered 11/10, 2012 at 8:10 Comment(0)
G
1

Heres a video i have created http://www.youtube.com/watch?v=HdtEQqu0yOY and below is a detailed explanation about the same.

Indexers helps to access contained collection with in a class using a simplified interface. It’s a syntactic sugar.

For instance lets say you have a customer class with addresses collection inside it. Now let’s say we would like to like fetch the addresses collection by “Pincode” and “PhoneNumber”. So the logical step would be that you would go and create two overloaded functions one which fetches by using “PhoneNumber” and the other by “PinCode”. You can see in the below code we have two functions defined.

Customer Customers = new Customer();
Customers.getAddress(1001);
Customers.getAddress("9090");

If you use indexer you can simplify the above code with something as shown in the below code.

Customer Customers = new Customer();
Address o = Customers[10001];
o = Customers["4320948"];

Cheers.

Goatfish answered 18/4, 2013 at 1:47 Comment(1)
IMHO the example isn't the best, here's why: Indexers must have the semantics of an array, so Customers[1001] would be expected to return a Customer and not another type like Address. In your case, you're better off using getAddress to convey the right intent. Otherwise, I would name the class implementing the indexer as Addresses and then fetch an Address as Addresses[1001]Nightrider
C
1

You can use an indexer to elegantly provide read/write multi-threading synchronization to a non-thread safe dictionary (or any non-thread safe collection):

internal class ThreadSafeIndexerClass
{
    public object this[int key]
    {
        get
        {
            // Aquire returns IDisposable and does Enter() Exit() on a certain ReaderWriterLockSlim instance
            using (_readLock.Aquire()) 
            {
                object subset;
                _dictionary.TryGetValue(key, out foundValue);
                return foundValue;
            }
        }
        set
        {
            // Aquire returns IDisposable and does Enter() Exit() on a certain ReaderWriterLockSlim instance
            using (_writeLock.Aquire())
                _dictionary[key] = value;
        }
    }
}

Useful especially when you don't want to use the heavyweight ConcurrentDictionary (or any other concurrent collection).

Calefactory answered 1/11, 2016 at 17:15 Comment(0)
L
0

http://code-kings.blogspot.in/2012/09/indexers-in-c-5.html

using System;

namespace Indexers_Example

{

class Indexers
{

    private Int16[] RollNumberVariable;

    public Indexers(Int16 size)
    {
        RollNumberVariable = new Int16[size];

        for (int i = 0; i < size; i++)
        {
            RollNumberVariable[i] = 0;
        }
    }

    public Int16 this[int pos]
    {
        get
        {
            return RollNumberVariable[pos];
        }
        set
        {
            RollNumberVariable[pos] = value;
        }
    }
}

}

Landsturm answered 11/10, 2012 at 8:16 Comment(0)
S
0

Addition to @code-kings post.

Moreover, calling RollNumberVariable[0] will trigger the behavior of the default collection's indexer. While indexers are actually properties, it's on your behalf to write your own logic when extracting data. You can easily delegate most of index-parameter value to the internal collection but you can also return any arbitrary value for certain index values.

Just an example - you can have 2+ internal collections in different format, but the external user will interact with them trough a single indexer (which will kinda work as a dispatcher), while those collection will be hidden. This pretty much encourages the encapsulation principle.

Sevier answered 11/10, 2012 at 8:30 Comment(0)
M
0

I am trying to get images from a sequence file. I need some sort of a 2D array or a jagged array to hold the pixel values. I am using indexers instead of arrays because looping over indexers is faster than looping over 2D or jagged arrays.

Macur answered 4/8, 2014 at 22:21 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.