C# Dictionary select performance benchmarks
These benchmarks are testing a Dictionary <int,string> and a  Dictionary<IUserKey, string>
In theory retrieval on a Hash Table (which is the data container inside each dictionary) should be constant regardless of number of items in the Hash Table. 
The reason being is that accessing a hash table is a random access of an array element  which of the key is generated by a Hashing algorithm.
1-      Dictionary.GetItem(key)
2-      f(Key) à indexValue: random access index of value. F is the hash algorithm.
3-      Dictionary.hashtable[indexValue] 
However, The problem is that the hash algorithm is not generating a unique index all the time, when it doesn’t it a collision event!
Collisions are the culprit of decrease in hash table item retrievals (long story short) 
In these set of tests I am examining the different items added to dictionary an int an object and a Object with specified HashCode algorithm.
DictionaryPerfTest10Mil1Mil means that Test was ran over a dictionary filled with 10 million items and it was queried 1 million times
Int dictionary-  
  private static void IntDictPerf(int fillSize, int selectionSize)
        {
            Random random = new Random();
            Dictionary<int, string> dictHash = new Dictionary<int, string>(fillSize);
            try
            {
                for (int i = 0; i < fillSize; i++)
                {
                    dictHash.Add(i, "value" + i.ToString());
                }
            }
            catch (Exception )
            {
                System.Diagnostics.Debug.WriteLine("Collision occured");
            }
            Stopwatch stopWatch = new Stopwatch();
            stopWatch.Start();
            for (int i = 0; i < selectionSize; i++)
            {
                int index = random.Next(fillSize);
                var x = dictHash[index];
            }
            stopWatch.Stop();
            TimeSpan ts = stopWatch.Elapsed;
            string elapsedTime = String.Format("{0:00}:{1:00}:{2:00}.{3:00}",
                ts.Hours, ts.Minutes, ts.Seconds, ts.Milliseconds / 10);
            System.Diagnostics.Debug.WriteLine(elapsedTime, "RunTime");
        }
Result:
Under 200K items fetch time is almost constant above 300K to 500K it doubles and above 500K it triples.
1st Run:
IntDictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
IntDictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.20
IntDictionaryPerfTest500K1Mil - RunTime: 00:00:00.15
IntDictionaryPerfTest300K1Mil - RunTime: 00:00:00.11
IntDictionaryPerfTest200K1Mil - RunTime: 00:00:00.06
IntDictionaryPerfTest100K1Mil - RunTime: 00:00:00.06
IntDictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
IntDictionaryPerfTest1K1Mil - RunTime: 00:00:00.04
2nd Run:
IntDictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
IntDictionaryPerfTest500K1Mil - RunTime: 00:00:00.15
IntDictionaryPerfTest300K1Mil - RunTime: 00:00:00.09
IntDictionaryPerfTest100K1Mil - RunTime: 00:00:00.05
IntDictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
IntDictionaryPerfTest1K1Mil - RunTime: 00:00:00.04
3rd Run:
IntDictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
IntDictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.21
IntDictionaryPerfTest500K1Mil - RunTime: 00:00:00.16
IntDictionaryPerfTest100K1Mil - RunTime: 00:00:00.05
IntDictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
4th Run:
DictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
DictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.20
DictionaryPerfTest100K1Mil - RunTime: 00:00:00.05
DictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
5th Run:
DictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
DictionaryPerfTest100K1Mil - RunTime: 00:00:00.05
DictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.21
DictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
6th Run:
DictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.27
DictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.21
DictionaryPerfTest100K1Mil - RunTime: 00:00:00.05
DictionaryPerfTest10K1Mil - RunTime: 00:00:00.05
DictionaryPerfTest1K1Mil - RunTime: 00:00:00.04
User Key struct - Default hashcode (GetHashCode())
public interface IUserKey
    {
        int ID { get; set; }
    }
    public struct UserKey : IUserKey
    {
        public int ID
        {
            get;
            set;
        }
       // public override int GetHashCode()
        //{
        //    return this.ID.GetHashCode();
        //}
    }
private static void UserKeyDictPerf(int fillSize, int selectionSize)
        {
            Random random = new Random();
            Dictionary<IUserKey, string> dictHash = new Dictionary<IUserKey, string>(fillSize);
            try
            {
                for (int i = 0; i < fillSize; i++)
                {
                    IUserKey usr = new UserKey() { ID=i };
                    dictHash.Add(usr, "value" + i.ToString());
                }
            }
            catch (Exception )
            {
                System.Diagnostics.Debug.WriteLine("Collision occured");
            }
            Stopwatch stopWatch = new Stopwatch();
            stopWatch.Start();
            for (int i = 0; i < selectionSize; i++)
            {
                //int index = random.Next(fillSize);
                IUserKey usr = new UserKey() { ID = random.Next(fillSize) };
                var x = dictHash[usr];
            }
            stopWatch.Stop();
            TimeSpan ts = stopWatch.Elapsed;
            string elapsedTime = String.Format("{0:00}:{1:00}:{2:00}.{3:00}",
                ts.Hours, ts.Minutes, ts.Seconds, ts.Milliseconds / 10);
            System.Diagnostics.Debug.WriteLine(elapsedTime, "RunTime");
        }
DictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.80
DictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.47
DictionaryPerfTest100K1Mil - RunTime: 00:00:00.29
DictionaryPerfTest10K1Mil - RunTime: 00:00:00.16
DictionaryPerfTest1K1Mil - RunTime: 00:00:00.14
User Key struct - Overriden hash code
Same as the above test with the exception  of defining the HashCode method
    public struct UserKey : IUserKey
    {
        public int ID
        {
            get;
            set;
        }
        public override int GetHashCode()
        {
            return this.ID.GetHashCode();
        }
    }
DictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.78
DictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.48
DictionaryPerfTest100K1Mil - RunTime: 00:00:00.30
DictionaryPerfTest10K1Mil - RunTime: 00:00:00.17
DictionaryPerfTest1K1Mil - RunTime: 00:00:00.15
User Lo Key struct –
UserLoKeyDictionaryPerfTest10Mil1Mil - RunTime: 00:00:00.79
UserLoKeyDictionaryPerfTest1Mil1Mil - RunTime: 00:00:00.48
UserLoKeyDictionaryPerfTest100K1Mil - RunTime: 00:00:00.28
UserLoKeyDictionaryPerfTest10K1Mil - RunTime: 00:00:00.16
UserLoKeyDictionaryPerfTest1K1Mil - RunTime: 00:00:00.15
 
