Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

C# dictionary - one key, many values

I want to create a data store to allow me to store some data.

The first idea was to create a dictionary where you have one key with many values, so a bit like a one-to-many relationship.

I think the dictionary only has one key value.

How else could I store this information?

like image 896
Marc G Avatar asked Jan 20 '10 11:01

Marc G


People also ask

What C is used for?

C programming language is a machine-independent programming language that is mainly used to create many types of applications and operating systems such as Windows, and other complicated programs such as the Oracle database, Git, Python interpreter, and games and is considered a programming foundation in the process of ...

Is C language easy?

C is a general-purpose language that most programmers learn before moving on to more complex languages. From Unix and Windows to Tic Tac Toe and Photoshop, several of the most commonly used applications today have been built on C. It is easy to learn because: A simple syntax with only 32 keywords.

What is the full name of C?

In the real sense it has no meaning or full form. It was developed by Dennis Ritchie and Ken Thompson at AT&T bell Lab. First, they used to call it as B language then later they made some improvement into it and renamed it as C and its superscript as C++ which was invented by Dr. Stroustroupe.

Is C programming hard?

C is more difficult to learn than JavaScript, but it's a valuable skill to have because most programming languages are actually implemented in C. This is because C is a “machine-level” language. So learning it will teach you how a computer works and will actually make learning new languages in the future easier.


2 Answers

As of .NET 3.5+, instead of using a Dictionary<IKey, List<IValue>>, you can use a Lookup from the LINQ namespace:

// Lookup Order by payment status (1:m) // would need something like Dictionary<Boolean, IEnumerable<Order>> orderIdByIsPayed ILookup<Boolean, Order> byPayment = orderList.ToLookup(o => o.IsPayed); IEnumerable<Order> payedOrders = byPayment[false]; 

From MSDN:

A Lookup<TKey, TElement> resembles a Dictionary<TKey, TValue>. The difference is that a Dictionary<TKey, TValue> maps keys to single values, whereas a Lookup<TKey, TElement> maps keys to collections of values.

You can create an instance of a Lookup<TKey, TElement> by calling ToLookup on an object that implements IEnumerable.

You may also want to read this answer to a related question. For more information, consult MSDN.

Full example:

using System; using System.Collections.Generic; using System.Linq;  namespace LinqLookupSpike {     class Program     {         static void Main(String[] args)         {             // Init             var orderList = new List<Order>();             orderList.Add(new Order(1, 1, 2010, true)); // (orderId, customerId, year, isPayed)             orderList.Add(new Order(2, 2, 2010, true));             orderList.Add(new Order(3, 1, 2010, true));             orderList.Add(new Order(4, 2, 2011, true));             orderList.Add(new Order(5, 2, 2011, false));             orderList.Add(new Order(6, 1, 2011, true));             orderList.Add(new Order(7, 3, 2012, false));              // Lookup Order by its id (1:1, so usual dictionary is ok)             Dictionary<Int32, Order> orders = orderList.ToDictionary(o => o.OrderId, o => o);              // Lookup Order by customer (1:n)             // would need something like Dictionary<Int32, IEnumerable<Order>> orderIdByCustomer             ILookup<Int32, Order> byCustomerId = orderList.ToLookup(o => o.CustomerId);             foreach (var customerOrders in byCustomerId)             {                 Console.WriteLine("Customer {0} ordered:", customerOrders.Key);                 foreach (var order in customerOrders)                 {                     Console.WriteLine("    Order {0} is payed: {1}", order.OrderId, order.IsPayed);                 }             }              // The same using old fashioned Dictionary             Dictionary<Int32, List<Order>> orderIdByCustomer;             orderIdByCustomer = byCustomerId.ToDictionary(g => g.Key, g => g.ToList());             foreach (var customerOrders in orderIdByCustomer)             {                 Console.WriteLine("Customer {0} ordered:", customerOrders.Key);                 foreach (var order in customerOrders.Value)                 {                     Console.WriteLine("    Order {0} is payed: {1}", order.OrderId, order.IsPayed);                 }             }              // Lookup Order by payment status (1:m)             // would need something like Dictionary<Boolean, IEnumerable<Order>> orderIdByIsPayed             ILookup<Boolean, Order> byPayment = orderList.ToLookup(o => o.IsPayed);             IEnumerable<Order> payedOrders = byPayment[false];             foreach (var payedOrder in payedOrders)             {                 Console.WriteLine("Order {0} from Customer {1} is not payed.", payedOrder.OrderId, payedOrder.CustomerId);             }         }          class Order         {             // Key properties             public Int32 OrderId { get; private set; }             public Int32 CustomerId { get; private set; }             public Int32 Year { get; private set; }             public Boolean IsPayed { get; private set; }              // Additional properties             // private List<OrderItem> _items;              public Order(Int32 orderId, Int32 customerId, Int32 year, Boolean isPayed)             {                 OrderId = orderId;                 CustomerId = customerId;                 Year = year;                 IsPayed = isPayed;             }         }     } } 

Remark on Immutability

By default, lookups are kind of immutable and accessing the internals would involve reflection. If you need mutability and don't want to write your own wrapper, you could use MultiValueDictionary (formerly known as MultiDictionary) from corefxlab (formerly part ofMicrosoft.Experimental.Collections which isn't updated anymore).

like image 166
mbx Avatar answered Oct 19 '22 23:10

mbx


You can use a list for the second generic type. For example a dictionary of strings keyed by a string:

Dictionary<string, List<string>> myDict; 
like image 32
Oded Avatar answered Oct 20 '22 00:10

Oded