Creating a C# Dictionary That Allows Duplicate Keys

Recently, I ran into an issue where I needed a Dictionary to make the application more performant, but there were duplicate keys. To get around this issue, I used System.Collection.Generic.KeyValuePair<TKey, TValue>. This allows one to create a quasi-Dictionary with similar performance and it allows duplicate keys.

We are going to build a simple Member class to track members who happen to all have the same last name, which is the key. Here is the class definition:

public class Member : IEqualityComparer<Member>
{
		public int Id { get; set; }
		public string FirstName { get; set; }
		public string LastName { get; set; }

		public bool Equals([DisallowNull] Member x, [DisallowNull] Member y)
		{
			if (x.LastName == y.LastName)
			{
				return true;
			}
			return false;
		}

		public int GetHashCode([DisallowNull] Member obj)
		{
			return new { obj.LastName }.GetHashCode();
		}
}

The class Member implements the IEqualityComparer<T> generic interface. That means it must implement the following methods:

public bool Equals(T x, T y)
public int GetHashCode(T obj)

As you can see in the implementation, I’m using the LastName field for comparison and for the hashcode.

Let’s write a console app that will use the Member class. We’ll create a list of Members that have the same last names, then create a Dictionary<TKey, TValue> and a KeyValuePair<TKey, TValue>, and insert data into those structures. If you’ve used Dictionary<TKey, TValue>, then you know it doesn’t allow duplicates. So, we’re going to use TryAdd<TKey, TValue> which returns a boolean if the add operation was successful.

using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;

namespace DictionaryWithDuplicates
{
	class Program
	{
		static void Main(string[] args)
		{
			List&lt;KeyValuePair&lt;string, Member>> MemberDictionary = new List&lt;KeyValuePair&lt;string, Member>>();
			Dictionary&lt;string, Member> BaseDictionary = new Dictionary&lt;string, Member>();

			List&lt;Member> members = new List&lt;Member>();
			members.Add(new Member
			{
				Id = 1,
				LastName = "Person",
				FirstName = "Test"
			});

			members.Add(new Member
			{
				Id = 2,
				LastName = "Person",
				FirstName = "Another"
			});
			
			members.Add(new Member
			{
				Id = 3,
				LastName = "Person",
				FirstName = "This"
			});
			
			foreach (var member in members)
			{
				MemberDictionary.Add(new KeyValuePair&lt;string, Member>(member.LastName, member));
				bool added = BaseDictionary.TryAdd&lt;string, Member>(member.LastName, member);
				if (added == false)
				{
					Console.WriteLine("Failed to add Id = " + member.Id + ": " + member.FirstName + ' ' + member.LastName);
				}
			}

			var output1 = MemberDictionary.Where(x => x.Key == "Person" &amp;&amp; x.Value.Id == 1).FirstOrDefault();
			var output2 = MemberDictionary.Where(x => x.Key == "Person" &amp;&amp; x.Value.Id == 2).FirstOrDefault();
			var output3 = MemberDictionary.Where(x => x.Key == "Person" &amp;&amp; x.Value.Id == 3).FirstOrDefault();

			Console.WriteLine("Id = " + output1.Value.Id + ": " + output1.Value.FirstName + ' ' + output1.Value.LastName);
			Console.WriteLine("Id = " + output2.Value.Id + ": " + output2.Value.FirstName + ' ' + output2.Value.LastName);
			Console.WriteLine("Id = " + output3.Value.Id + ": " + output3.Value.FirstName + ' ' + output3.Value.LastName);
		}
	}

	public class Member : IEqualityComparer&lt;Member>
	{
		public int Id { get; set; }
		public string FirstName { get; set; }
		public string LastName { get; set; }
		public bool Equals([DisallowNull] Member x, [DisallowNull] Member y)
		{
			if (x.LastName == y.LastName)
			{
				return true;
			}
			return false;
		}
		public int GetHashCode([DisallowNull] Member obj)
		{
			return new { obj.LastName }.GetHashCode();
		}
	}
}

Run the application to see the results.

About the Author

Leave a Reply

This site uses Akismet to reduce spam. Learn how your comment data is processed.

You may also like these