Pourquoi n'y a-t-il pas d'implémentation générique de OrderedDictionary dans .net?

26

Pourquoi Microsoft n'a-t-il pas fourni d'implémentation générique de OrderedDictionary?

Il y a quelques implémentations personnalisées que j'ai vues, y compris: http://www.codeproject.com/KB/recipes/GenericOrderedDictionary.aspx

Mais pourquoi Microsoft ne l'a-t-il pas inclus dans la bibliothèque .net de base? Ils avaient sûrement une raison de ne pas construire un générique ... mais qu'est-ce que c'est?

Avant de poster ce message, j'ai vu: /programming/2629027/no-generic-implementation-of-ordereddictionary

Mais cela confirme simplement qu'il n'existe pas. Pas pourquoi ça n'existe pas.

Merci

nonot1
la source
2
Il y a toujours SortedDictionary<TKey, TValue>: msdn.microsoft.com/en-us/library/f7fta44c.aspx
Travis Gockel
2
À moins que je ne comprenne mal les documents SortedDict, ce n'est pas ce que je veux. Je ne veux pas faire de tri. Je veux juste un tableau auquel je peux également accéder par clé. Quoi qu'il en soit, je me demande vraiment pourquoi MS a ignoré cela. (Problèmes subtils, etc.)
nonot1
Quel est un cas d'utilisation typique pour un dictionnaire ordonné? J'ai du mal à penser à celui du haut de ma tête.
Carson63000
1
@Carson chaque fois que vous disposez d'un tableau d'éléments de données auquel vous avez également besoin d'un accès aléatoire rapide. Stocker simplement dans un Dict perd les informations de commande, et l'utilisation d'un tableau nécessite que vous mainteniez votre propre index.
nonot1
2
Demandez à Eric Lippert de lire et de répondre à votre question. Il est généralement très agréable à aider avec ce type de question. Je pense que vous pouvez le contacter via son blog: blogs.msdn.com/b/ericlippert
devuxer

Réponses:

17

En C # 4.0 En bref, j'ai lu ceci:

  1. An OrderedDictionaryest une combinaison de a HashTableetArrayList
  2. Il n'y a pas de générique ArrayList
  3. "La ArrayListclasse non générique est principalement utilisée pour la compatibilité descendante avec Framework 1.x ..."
  4. "An ArrayListest fonctionnellement similaire à List<object>"
  5. "La réflexion est plus facile avec un non générique ArrayListqu'avec un List<object>"

Conclusion?

Aucun générique OrderedDictionaryparce que sa construction sous-jacente est une classe dépréciée (officieusement) qui n'a pas de version générique elle-même.

radarbob
la source
4
Cela ne répond pas vraiment à la question. Un OrderedDictionary générique pourrait évidemment être construit sur un dictionnaire générique et une liste générique.
JacquesB
Un précepte d'utilisation de classe est si l'interface utilisateur ne change pas qui se soucie de la mise en œuvre? Eh bien, l'équipe de conception / compilateur de langage, disons, pourrait préférer le générique héritant de son homologue non générique . Mes sens sournois stupides. Pour commencer: les différents ancêtres (immédiats) sont-ils une mine terrestre quelque part sur le chemin de l'évolution de la OrderedDictionarycovariance et de la contre-variance?
radarbob
15

La OrderedDictionarysurcharge de l'opération d'indexation de sorte que l'indexation avec un entier Nmettra l'élément en position N, tandis que l'indexation avec un Objectrécupérera l'élément correspondant à cet objet. Si l'on devait créer un OrderedDictionary<int, string>appel myDictet ajouter des éléments (1, "George") et (0, "Fred") dans cet ordre, devrait myDict[0]retourner "George" ou "Fred"?

Un tel problème aurait pu être résolu en imposant une contrainte de classe au type de clé. D'un autre côté, une grande partie de l'utilité des collections génériques découle de leur capacité à travailler efficacement avec les types de valeur. Imposer une contrainte de classe au type de clé semble un peu moche.

Si la classe n'avait pas à être conforme CLS mais devait simplement travailler avec vb.net, une conception judicieuse aurait pu être d'utiliser des propriétés indexées nommées. Ainsi, dans l'exemple ci-dessus, myDict.ByKey[0]aurait donné "Fred" et myDict.BySequence[0]aurait donné "George". Malheureusement, les langages comme C # ne prennent pas en charge les propriétés indexées nommées. Alors que l'on aurait pu claquer quelque chose pour permettre l'utilisation de la syntaxe ci-dessus même sans ces propriétés, la décision malheureuse d'envelopper les champs de structures comme Pointet Rectanglesignifie que pour myDict.ByKey[0] = "Wally"fonctionner, myDict.ByKeydevrait renvoyer un nouvel objet de classe. Une structure serait plus efficace, mais les compilateurs rejetteraient ce qui ressemblait à une écriture dans une structure en lecture seule (bien que la propriété ne modifie pas la structure renvoyée parByKey, mais modifiez plutôt la collection à laquelle il contient une référence).

Personnellement, je pense qu'un objet dictionnaire-ish qui a été spécifié comme gardant la trace de l'ordre d'insertion serait une bonne chose à avoir; Je voudrais également avoir un objet dictionnaire-ish qui pourrait facilement retourner la clé associée à une clé particulière (de sorte que, par exemple, si l'on a un dictionnaire qui ne respecte pas la casse et a ajouté un enregistrement avec une clé de "GEORGE", un pourrait demander au dictionnaire quelle clé est associée à "George" sans avoir à parcourir tous les KeyValuePairobjets retournés dans une énumération.

supercat
la source
3

Parce que le maintien de l'ordre empêche la recherche O (1) qu'IDictionary implique, sauf si vous encapsulez deux collections (une pour la commande, une pour la recherche), ce qui rend l'ajout / la suppression moins performant et augmente l'utilisation de la mémoire. Ou vous pourriez avoir une recherche plus lente en échange d'une utilisation moindre de la mémoire.

Je suppose qu'il n'y avait pas de choix «clairement meilleur» ici, donc il n'est pas entré dans la bibliothèque standard. Surtout vers 2.0, C # apprenait encore des erreurs de Java. Je ne serais pas surpris si l'approche `` tout et l'évier de cuisine '' de Java pour les collections dans leur bibliothèque standard était également considérée comme quelque chose à éviter.

Telastyn
la source
1
Le fait d'avoir une recherche plus lente en échange de moins de mémoire rend le OrderedDictionarya List. J'imagine que toute personne ayant un cas d'utilisation légitime pour un l' OrderedDictionaryutilise spécifiquement parce qu'elle a besoin d'une classe de collection qui a la recherche O (1) mais se souvient également de l'ordre d'insertion, auquel cas l'utilisation de mémoire supplémentaire est inévitable et donc acceptable.
Abion47