অর্ডারডেরিয়ার কোনও জেনেরিক প্রয়োগ নেই?


136

.NET 3.5 OrderedDictionaryতে (যা System.Collections.Specializedনামস্থানটিতে রয়েছে) এর জেনেরিক বাস্তবায়ন বলে মনে হয় না । এখানে কি আমি মিস করছি?

কার্যকারিতা সরবরাহ করার জন্য আমি সেখানে বাস্তবায়ন খুঁজে পেয়েছি, তবে ভাবছি কিনা / কেন বাক্সের বাইরে জেনেরিক বাস্তবায়ন হয় না এবং যদি কেউ জানে যে এটি নেট .0 এ আছে কি না?


1
এখানে একটি এর বাস্তবায়ন রয়েছে OrderedDictionary<T>: কোডেপ্রজেক্ট
টিম শেমলেটার


OrderedDictionary <টি> আমার বাস্তবায়ন হে রয়েছে (1) ঢোকান / মুছতে কারণ এটি ArrayList পরিবর্তে একটি LinkedList ব্যবহার সন্নিবেশ শৃঙ্খলা বজায় রাখার জন্য: clintonbrennan.com/2013/12/...
ক্লিনটন

2
আপনার যদি কেবল এন্ট্রিগুলিকে যুক্ত করা ক্রম অনুসারে পুনরাবৃত্তি করতে সক্ষম হয় তবে তালিকা <কীভ্যালিউপায়ার <টি কে, টিভিয়াল >> তালিকাটি যথেষ্ট ভাল হতে পারে। (অনুমোদিত, কোনও সাধারণ সমাধান নয়, তবে কিছু উদ্দেশ্যে যথেষ্ট ভাল))
yoyo

1
এটি একটি দুর্ভাগ্যজনক বাদ পড়ে। অন্যান্য ভাল ডাটা টাইপ আছে Systems.Collections.Generic। আসুন OrderedDictionary<TKey,TValue>নেট। 5- র জন্য অনুরোধ করুন অন্যরা যেমন উল্লেখ করেছে, কেসটি কীটি হ'ল অবক্ষয়যুক্ত, এবং তার জন্য বিশেষ যত্নের প্রয়োজন হবে।
কর্নেল আতঙ্ক

উত্তর:


60

তুমি ঠিক বলছো. OrderedDictionaryকাঠামোর মধ্যে কোনও জেনেরিক সমতুল্য নেই ।

(এটি এখনও নেট নেট 4 এর ক্ষেত্রে, আমি যতদূর অবগত আছি))

তবে আপনি ভিজ্যুয়াল স্টুডিওর ইউজারভয়েসে (2016-10-04) এটির পক্ষে ভোট দিতে পারেন !


95

জেনেরিক প্রয়োগ OrderedDictionaryকার্যকরভাবে অসুবিধাজনক নয়, তবে এটি অযথা সময় সাপেক্ষ এবং সত্যই এই ক্লাসটি মাইক্রোসফ্টের পক্ষ থেকে একটি বিশাল নজরদারি। এটি বাস্তবায়নের একাধিক উপায় রয়েছে তবে আমি KeyedCollectionআমার অভ্যন্তরীণ স্টোরেজটির জন্য একটি ব্যবহার পছন্দ করেছি। এই পদ্ধতিতে বাছাই করার জন্য আমি বিভিন্ন পদ্ধতি প্রয়োগ করতেও বেছে নিয়েছি List<T>কারণ এটি মূলত একটি হাইব্রিড আইলিস্ট এবং আইডিকোরিয়াল। আমি আমার বাস্তবায়ন এখানে উত্তরোত্তর জন্য অন্তর্ভুক্ত করেছি।

এখানে ইন্টারফেস। লক্ষ্য করুন যে এতে অন্তর্ভুক্ত রয়েছে System.Collections.Specialized.IOrderedDictionary, যা মাইক্রোসফ্ট সরবরাহ করেছিল এই ইন্টারফেসের অ জেনারিক সংস্করণ।

// http://unlicense.org
using System;
using System.Collections.Generic;
using System.Collections.Specialized;

namespace mattmc3.Common.Collections.Generic {

    public interface IOrderedDictionary<TKey, TValue> : IDictionary<TKey, TValue>, IOrderedDictionary {
        new TValue this[int index] { get; set; }
        new TValue this[TKey key] { get; set; }
        new int Count { get; }
        new ICollection<TKey> Keys { get; }
        new ICollection<TValue> Values { get; }
        new void Add(TKey key, TValue value);
        new void Clear();
        void Insert(int index, TKey key, TValue value);
        int IndexOf(TKey key);
        bool ContainsValue(TValue value);
        bool ContainsValue(TValue value, IEqualityComparer<TValue> comparer);
        new bool ContainsKey(TKey key);
        new IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator();
        new bool Remove(TKey key);
        new void RemoveAt(int index);
        new bool TryGetValue(TKey key, out TValue value);
        TValue GetValue(TKey key);
        void SetValue(TKey key, TValue value);
        KeyValuePair<TKey, TValue> GetItem(int index);
        void SetItem(int index, TValue value);
    }

}

সহায়ক শ্রেণীর পাশাপাশি বাস্তবায়ন এখানে:

// http://unlicense.org
using System;
using System.Collections.ObjectModel;
using System.Diagnostics;
using System.Collections;
using System.Collections.Specialized;
using System.Collections.Generic;
using System.Linq;

namespace mattmc3.Common.Collections.Generic {

    /// <summary>
    /// A dictionary object that allows rapid hash lookups using keys, but also
    /// maintains the key insertion order so that values can be retrieved by
    /// key index.
    /// </summary>
    public class OrderedDictionary<TKey, TValue> : IOrderedDictionary<TKey, TValue> {

        #region Fields/Properties

        private KeyedCollection2<TKey, KeyValuePair<TKey, TValue>> _keyedCollection;

        /// <summary>
        /// Gets or sets the value associated with the specified key.
        /// </summary>
        /// <param name="key">The key associated with the value to get or set.</param>
        public TValue this[TKey key] {
            get {
                return GetValue(key);
            }
            set {
                SetValue(key, value);
            }
        }

        /// <summary>
        /// Gets or sets the value at the specified index.
        /// </summary>
        /// <param name="index">The index of the value to get or set.</param>
        public TValue this[int index] {
            get {
                return GetItem(index).Value;
            }
            set {
                SetItem(index, value);
            }
        }

        public int Count {
            get { return _keyedCollection.Count; }
        }

        public ICollection<TKey> Keys {
            get {
                return _keyedCollection.Select(x => x.Key).ToList();
            }
        }

        public ICollection<TValue> Values {
            get {
                return _keyedCollection.Select(x => x.Value).ToList();
            }
        }

        public IEqualityComparer<TKey> Comparer {
            get;
            private set;
        }

        #endregion

        #region Constructors

        public OrderedDictionary() {
            Initialize();
        }

        public OrderedDictionary(IEqualityComparer<TKey> comparer) {
            Initialize(comparer);
        }

        public OrderedDictionary(IOrderedDictionary<TKey, TValue> dictionary) {
            Initialize();
            foreach (KeyValuePair<TKey, TValue> pair in dictionary) {
                _keyedCollection.Add(pair);
            }
        }

        public OrderedDictionary(IOrderedDictionary<TKey, TValue> dictionary, IEqualityComparer<TKey> comparer) {
            Initialize(comparer);
            foreach (KeyValuePair<TKey, TValue> pair in dictionary) {
                _keyedCollection.Add(pair);
            }
        }

        #endregion

        #region Methods

        private void Initialize(IEqualityComparer<TKey> comparer = null) {
            this.Comparer = comparer;
            if (comparer != null) {
                _keyedCollection = new KeyedCollection2<TKey, KeyValuePair<TKey, TValue>>(x => x.Key, comparer);
            }
            else {
                _keyedCollection = new KeyedCollection2<TKey, KeyValuePair<TKey, TValue>>(x => x.Key);
            }
        }

        public void Add(TKey key, TValue value) {
            _keyedCollection.Add(new KeyValuePair<TKey, TValue>(key, value));
        }

        public void Clear() {
            _keyedCollection.Clear();
        }

        public void Insert(int index, TKey key, TValue value) {
            _keyedCollection.Insert(index, new KeyValuePair<TKey, TValue>(key, value));
        }

        public int IndexOf(TKey key) {
            if (_keyedCollection.Contains(key)) {
                return _keyedCollection.IndexOf(_keyedCollection[key]);
            }
            else {
                return -1;
            }
        }

        public bool ContainsValue(TValue value) {
            return this.Values.Contains(value);
        }

        public bool ContainsValue(TValue value, IEqualityComparer<TValue> comparer) {
            return this.Values.Contains(value, comparer);
        }

        public bool ContainsKey(TKey key) {
            return _keyedCollection.Contains(key);
        }

        public KeyValuePair<TKey, TValue> GetItem(int index) {
            if (index < 0 || index >= _keyedCollection.Count) {
                throw new ArgumentException(String.Format("The index was outside the bounds of the dictionary: {0}", index));
            }
            return _keyedCollection[index];
        }

        /// <summary>
        /// Sets the value at the index specified.
        /// </summary>
        /// <param name="index">The index of the value desired</param>
        /// <param name="value">The value to set</param>
        /// <exception cref="ArgumentOutOfRangeException">
        /// Thrown when the index specified does not refer to a KeyValuePair in this object
        /// </exception>
        public void SetItem(int index, TValue value) {
            if (index < 0 || index >= _keyedCollection.Count) {
                throw new ArgumentException("The index is outside the bounds of the dictionary: {0}".FormatWith(index));
            }
            var kvp = new KeyValuePair<TKey, TValue>(_keyedCollection[index].Key, value);
            _keyedCollection[index] = kvp;
        }

        public IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator() {
            return _keyedCollection.GetEnumerator();
        }

        public bool Remove(TKey key) {
            return _keyedCollection.Remove(key);
        }

        public void RemoveAt(int index) {
            if (index < 0 || index >= _keyedCollection.Count) {
                throw new ArgumentException(String.Format("The index was outside the bounds of the dictionary: {0}", index));
            }
            _keyedCollection.RemoveAt(index);
        }

        /// <summary>
        /// Gets the value associated with the specified key.
        /// </summary>
        /// <param name="key">The key associated with the value to get.</param>
        public TValue GetValue(TKey key) {
            if (_keyedCollection.Contains(key) == false) {
                throw new ArgumentException("The given key is not present in the dictionary: {0}".FormatWith(key));
            }
            var kvp = _keyedCollection[key];
            return kvp.Value;
        }

        /// <summary>
        /// Sets the value associated with the specified key.
        /// </summary>
        /// <param name="key">The key associated with the value to set.</param>
        /// <param name="value">The the value to set.</param>
        public void SetValue(TKey key, TValue value) {
            var kvp = new KeyValuePair<TKey, TValue>(key, value);
            var idx = IndexOf(key);
            if (idx > -1) {
                _keyedCollection[idx] = kvp;
            }
            else {
                _keyedCollection.Add(kvp);
            }
        }

        public bool TryGetValue(TKey key, out TValue value) {
            if (_keyedCollection.Contains(key)) {
                value = _keyedCollection[key].Value;
                return true;
            }
            else {
                value = default(TValue);
                return false;
            }
        }

        #endregion

        #region sorting
        public void SortKeys() {
            _keyedCollection.SortByKeys();
        }

        public void SortKeys(IComparer<TKey> comparer) {
            _keyedCollection.SortByKeys(comparer);
        }

        public void SortKeys(Comparison<TKey> comparison) {
            _keyedCollection.SortByKeys(comparison);
        }

        public void SortValues() {
            var comparer = Comparer<TValue>.Default;
            SortValues(comparer);
        }

        public void SortValues(IComparer<TValue> comparer) {
            _keyedCollection.Sort((x, y) => comparer.Compare(x.Value, y.Value));
        }

        public void SortValues(Comparison<TValue> comparison) {
            _keyedCollection.Sort((x, y) => comparison(x.Value, y.Value));
        }
        #endregion

        #region IDictionary<TKey, TValue>

        void IDictionary<TKey, TValue>.Add(TKey key, TValue value) {
            Add(key, value);
        }

        bool IDictionary<TKey, TValue>.ContainsKey(TKey key) {
            return ContainsKey(key);
        }

        ICollection<TKey> IDictionary<TKey, TValue>.Keys {
            get { return Keys; }
        }

        bool IDictionary<TKey, TValue>.Remove(TKey key) {
            return Remove(key);
        }

        bool IDictionary<TKey, TValue>.TryGetValue(TKey key, out TValue value) {
            return TryGetValue(key, out value);
        }

        ICollection<TValue> IDictionary<TKey, TValue>.Values {
            get { return Values; }
        }

        TValue IDictionary<TKey, TValue>.this[TKey key] {
            get {
                return this[key];
            }
            set {
                this[key] = value;
            }
        }

        #endregion

        #region ICollection<KeyValuePair<TKey, TValue>>

        void ICollection<KeyValuePair<TKey, TValue>>.Add(KeyValuePair<TKey, TValue> item) {
            _keyedCollection.Add(item);
        }

        void ICollection<KeyValuePair<TKey, TValue>>.Clear() {
            _keyedCollection.Clear();
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.Contains(KeyValuePair<TKey, TValue> item) {
            return _keyedCollection.Contains(item);
        }

        void ICollection<KeyValuePair<TKey, TValue>>.CopyTo(KeyValuePair<TKey, TValue>[] array, int arrayIndex) {
            _keyedCollection.CopyTo(array, arrayIndex);
        }

        int ICollection<KeyValuePair<TKey, TValue>>.Count {
            get { return _keyedCollection.Count; }
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.IsReadOnly {
            get { return false; }
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.Remove(KeyValuePair<TKey, TValue> item) {
            return _keyedCollection.Remove(item);
        }

        #endregion

        #region IEnumerable<KeyValuePair<TKey, TValue>>

        IEnumerator<KeyValuePair<TKey, TValue>> IEnumerable<KeyValuePair<TKey, TValue>>.GetEnumerator() {
            return GetEnumerator();
        }

        #endregion

        #region IEnumerable

        IEnumerator IEnumerable.GetEnumerator() {
            return GetEnumerator();
        }

        #endregion

        #region IOrderedDictionary

        IDictionaryEnumerator IOrderedDictionary.GetEnumerator() {
            return new DictionaryEnumerator<TKey, TValue>(this);
        }

        void IOrderedDictionary.Insert(int index, object key, object value) {
            Insert(index, (TKey)key, (TValue)value);
        }

        void IOrderedDictionary.RemoveAt(int index) {
            RemoveAt(index);
        }

        object IOrderedDictionary.this[int index] {
            get {
                return this[index];
            }
            set {
                this[index] = (TValue)value;
            }
        }

        #endregion

        #region IDictionary

        void IDictionary.Add(object key, object value) {
            Add((TKey)key, (TValue)value);
        }

        void IDictionary.Clear() {
            Clear();
        }

        bool IDictionary.Contains(object key) {
            return _keyedCollection.Contains((TKey)key);
        }

        IDictionaryEnumerator IDictionary.GetEnumerator() {
            return new DictionaryEnumerator<TKey, TValue>(this);
        }

        bool IDictionary.IsFixedSize {
            get { return false; }
        }

        bool IDictionary.IsReadOnly {
            get { return false; }
        }

        ICollection IDictionary.Keys {
            get { return (ICollection)this.Keys; }
        }

        void IDictionary.Remove(object key) {
            Remove((TKey)key);
        }

        ICollection IDictionary.Values {
            get { return (ICollection)this.Values; }
        }

        object IDictionary.this[object key] {
            get {
                return this[(TKey)key];
            }
            set {
                this[(TKey)key] = (TValue)value;
            }
        }

        #endregion

        #region ICollection

        void ICollection.CopyTo(Array array, int index) {
            ((ICollection)_keyedCollection).CopyTo(array, index);
        }

        int ICollection.Count {
            get { return ((ICollection)_keyedCollection).Count; }
        }

        bool ICollection.IsSynchronized {
            get { return ((ICollection)_keyedCollection).IsSynchronized; }
        }

        object ICollection.SyncRoot {
            get { return ((ICollection)_keyedCollection).SyncRoot; }
        }

        #endregion
    }

    public class KeyedCollection2<TKey, TItem> : KeyedCollection<TKey, TItem> {
        private const string DelegateNullExceptionMessage = "Delegate passed cannot be null";
        private Func<TItem, TKey> _getKeyForItemDelegate;

        public KeyedCollection2(Func<TItem, TKey> getKeyForItemDelegate)
            : base() {
            if (getKeyForItemDelegate == null) throw new ArgumentNullException(DelegateNullExceptionMessage);
            _getKeyForItemDelegate = getKeyForItemDelegate;
        }

        public KeyedCollection2(Func<TItem, TKey> getKeyForItemDelegate, IEqualityComparer<TKey> comparer)
            : base(comparer) {
            if (getKeyForItemDelegate == null) throw new ArgumentNullException(DelegateNullExceptionMessage);
            _getKeyForItemDelegate = getKeyForItemDelegate;
        }

        protected override TKey GetKeyForItem(TItem item) {
            return _getKeyForItemDelegate(item);
        }

        public void SortByKeys() {
            var comparer = Comparer<TKey>.Default;
            SortByKeys(comparer);
        }

        public void SortByKeys(IComparer<TKey> keyComparer) {
            var comparer = new Comparer2<TItem>((x, y) => keyComparer.Compare(GetKeyForItem(x), GetKeyForItem(y)));
            Sort(comparer);
        }

        public void SortByKeys(Comparison<TKey> keyComparison) {
            var comparer = new Comparer2<TItem>((x, y) => keyComparison(GetKeyForItem(x), GetKeyForItem(y)));
            Sort(comparer);
        }

        public void Sort() {
            var comparer = Comparer<TItem>.Default;
            Sort(comparer);
        }

        public void Sort(Comparison<TItem> comparison) {
            var newComparer = new Comparer2<TItem>((x, y) => comparison(x, y));
            Sort(newComparer);
        }

        public void Sort(IComparer<TItem> comparer) {
            List<TItem> list = base.Items as List<TItem>;
            if (list != null) {
                list.Sort(comparer);
            }
        }
    }

    public class Comparer2<T> : Comparer<T> {
        //private readonly Func<T, T, int> _compareFunction;
        private readonly Comparison<T> _compareFunction;

        #region Constructors

        public Comparer2(Comparison<T> comparison) {
            if (comparison == null) throw new ArgumentNullException("comparison");
            _compareFunction = comparison;
        }

        #endregion

        public override int Compare(T arg1, T arg2) {
            return _compareFunction(arg1, arg2);
        }
    }

    public class DictionaryEnumerator<TKey, TValue> : IDictionaryEnumerator, IDisposable {
        readonly IEnumerator<KeyValuePair<TKey, TValue>> impl;
        public void Dispose() { impl.Dispose(); }
        public DictionaryEnumerator(IDictionary<TKey, TValue> value) {
            this.impl = value.GetEnumerator();
        }
        public void Reset() { impl.Reset(); }
        public bool MoveNext() { return impl.MoveNext(); }
        public DictionaryEntry Entry {
            get {
                var pair = impl.Current;
                return new DictionaryEntry(pair.Key, pair.Value);
            }
        }
        public object Key { get { return impl.Current.Key; } }
        public object Value { get { return impl.Current.Value; } }
        public object Current { get { return Entry; } }
    }
}

এবং কয়েকটি পরীক্ষা ছাড়াই কোনও বাস্তবায়ন সম্পূর্ণ হবে না (তবে দুঃখজনকভাবে, এসও আমাকে এত পোস্টে একটি পোস্টে পোস্ট করতে দেবে না), তাই আপনাকে আপনার পরীক্ষাগুলি লেখার জন্য ছেড়ে যেতে হবে। তবে, আমি তাদের মধ্যে কয়েকটি রেখেছি যাতে আপনি এটি কীভাবে কাজ করে তা সম্পর্কে ধারণা পেতে পারেন:

// http://unlicense.org
using System;
using System.Collections.Generic;
using System.Linq;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using mattmc3.Common.Collections.Generic;

namespace mattmc3.Tests.Common.Collections.Generic {
    [TestClass]
    public class OrderedDictionaryTests {

        private OrderedDictionary<string, string> GetAlphabetDictionary(IEqualityComparer<string> comparer = null) {
            OrderedDictionary<string, string> alphabet = (comparer == null ? new OrderedDictionary<string, string>() : new OrderedDictionary<string, string>(comparer));
            for (var a = Convert.ToInt32('a'); a <= Convert.ToInt32('z'); a++) {
                var c = Convert.ToChar(a);
                alphabet.Add(c.ToString(), c.ToString().ToUpper());
            }
            Assert.AreEqual(26, alphabet.Count);
            return alphabet;
        }

        private List<KeyValuePair<string, string>> GetAlphabetList() {
            var alphabet = new List<KeyValuePair<string, string>>();
            for (var a = Convert.ToInt32('a'); a <= Convert.ToInt32('z'); a++) {
                var c = Convert.ToChar(a);
                alphabet.Add(new KeyValuePair<string, string>(c.ToString(), c.ToString().ToUpper()));
            }
            Assert.AreEqual(26, alphabet.Count);
            return alphabet;
        }

        [TestMethod]
        public void TestAdd() {
            var od = new OrderedDictionary<string, string>();
            Assert.AreEqual(0, od.Count);
            Assert.AreEqual(-1, od.IndexOf("foo"));

            od.Add("foo", "bar");
            Assert.AreEqual(1, od.Count);
            Assert.AreEqual(0, od.IndexOf("foo"));
            Assert.AreEqual(od[0], "bar");
            Assert.AreEqual(od["foo"], "bar");
            Assert.AreEqual(od.GetItem(0).Key, "foo");
            Assert.AreEqual(od.GetItem(0).Value, "bar");
        }

        [TestMethod]
        public void TestRemove() {
            var od = new OrderedDictionary<string, string>();

            od.Add("foo", "bar");
            Assert.AreEqual(1, od.Count);

            od.Remove("foo");
            Assert.AreEqual(0, od.Count);
        }

        [TestMethod]
        public void TestRemoveAt() {
            var od = new OrderedDictionary<string, string>();

            od.Add("foo", "bar");
            Assert.AreEqual(1, od.Count);

            od.RemoveAt(0);
            Assert.AreEqual(0, od.Count);
        }

        [TestMethod]
        public void TestClear() {
            var od = GetAlphabetDictionary();
            Assert.AreEqual(26, od.Count);
            od.Clear();
            Assert.AreEqual(0, od.Count);
        }

        [TestMethod]
        public void TestOrderIsPreserved() {
            var alphabetDict = GetAlphabetDictionary();
            var alphabetList = GetAlphabetList();
            Assert.AreEqual(26, alphabetDict.Count);
            Assert.AreEqual(26, alphabetList.Count);

            var keys = alphabetDict.Keys.ToList();
            var values = alphabetDict.Values.ToList();

            for (var i = 0; i < 26; i++) {
                var dictItem = alphabetDict.GetItem(i);
                var listItem = alphabetList[i];
                var key = keys[i];
                var value = values[i];

                Assert.AreEqual(dictItem, listItem);
                Assert.AreEqual(key, listItem.Key);
                Assert.AreEqual(value, listItem.Value);
            }
        }

        [TestMethod]
        public void TestTryGetValue() {
            var alphabetDict = GetAlphabetDictionary();
            string result = null;
            Assert.IsFalse(alphabetDict.TryGetValue("abc", out result));
            Assert.IsNull(result);
            Assert.IsTrue(alphabetDict.TryGetValue("z", out result));
            Assert.AreEqual("Z", result);
        }

        [TestMethod]
        public void TestEnumerator() {
            var alphabetDict = GetAlphabetDictionary();

            var keys = alphabetDict.Keys.ToList();
            Assert.AreEqual(26, keys.Count);

            var i = 0;
            foreach (var kvp in alphabetDict) {
                var value = alphabetDict[kvp.Key];
                Assert.AreEqual(kvp.Value, value);
                i++;
            }
        }

        [TestMethod]
        public void TestInvalidIndex() {
            var alphabetDict = GetAlphabetDictionary();
            try {
                var notGonnaWork = alphabetDict[100];
                Assert.IsTrue(false, "Exception should have thrown");
            }
            catch (Exception ex) {
                Assert.IsTrue(ex.Message.Contains("index is outside the bounds"));
            }
        }

        [TestMethod]
        public void TestMissingKey() {
            var alphabetDict = GetAlphabetDictionary();
            try {
                var notGonnaWork = alphabetDict["abc"];
                Assert.IsTrue(false, "Exception should have thrown");
            }
            catch (Exception ex) {
                Assert.IsTrue(ex.Message.Contains("key is not present"));
            }
        }

        [TestMethod]
        public void TestUpdateExistingValue() {
            var alphabetDict = GetAlphabetDictionary();
            Assert.IsTrue(alphabetDict.ContainsKey("c"));
            Assert.AreEqual(2, alphabetDict.IndexOf("c"));
            Assert.AreEqual(alphabetDict[2], "C");
            alphabetDict[2] = "CCC";
            Assert.IsTrue(alphabetDict.ContainsKey("c"));
            Assert.AreEqual(2, alphabetDict.IndexOf("c"));
            Assert.AreEqual(alphabetDict[2], "CCC");
        }

        [TestMethod]
        public void TestInsertValue() {
            var alphabetDict = GetAlphabetDictionary();
            Assert.IsTrue(alphabetDict.ContainsKey("c"));
            Assert.AreEqual(2, alphabetDict.IndexOf("c"));
            Assert.AreEqual(alphabetDict[2], "C");
            Assert.AreEqual(26, alphabetDict.Count);
            Assert.IsFalse(alphabetDict.ContainsValue("ABC"));

            alphabetDict.Insert(2, "abc", "ABC");
            Assert.IsTrue(alphabetDict.ContainsKey("c"));
            Assert.AreEqual(2, alphabetDict.IndexOf("abc"));
            Assert.AreEqual(alphabetDict[2], "ABC");
            Assert.AreEqual(27, alphabetDict.Count);
            Assert.IsTrue(alphabetDict.ContainsValue("ABC"));
        }

        [TestMethod]
        public void TestValueComparer() {
            var alphabetDict = GetAlphabetDictionary();
            Assert.IsFalse(alphabetDict.ContainsValue("a"));
            Assert.IsTrue(alphabetDict.ContainsValue("a", StringComparer.OrdinalIgnoreCase));
        }

        [TestMethod]
        public void TestSortByKeys() {
            var alphabetDict = GetAlphabetDictionary();
            var reverseAlphabetDict = GetAlphabetDictionary();
            Comparison<string> stringReverse = ((x, y) => (String.Equals(x, y) ? 0 : String.Compare(x, y) >= 1 ? -1 : 1));
            reverseAlphabetDict.SortKeys(stringReverse);
            for (int j = 0, k = 25; j < alphabetDict.Count; j++, k--) {
                var ascValue = alphabetDict.GetItem(j);
                var dscValue = reverseAlphabetDict.GetItem(k);
                Assert.AreEqual(ascValue.Key, dscValue.Key);
                Assert.AreEqual(ascValue.Value, dscValue.Value);
            }
        }

-- হালনাগাদ --

এটির জন্য উত্স এবং অন্যান্য সত্যই দরকারী অনুপস্থিত কোর। নেট গ্রন্থাগারগুলি এখানে: https://github.com/mattmc3/dotmore/blob/master/dotmore/ संग्रह / জেনেরিক / অর্ডারডডিয়োরিসি।


6
সর্বজনীন ডোমেন দ্বারা, আপনি কি এটি জিজ্ঞাসা করছেন যদি আপনি এটি ব্যবহার করতে পারেন, এটি সংশোধন করতে পারেন এবং এটি আপনার যেমন উদ্বেগ মুক্ত বলে মনে করেন - হ্যাঁ। নিঃসঙ্কোচে. যদি আপনি এটি লাইসেন্স হিসাবে বোঝাতে চান এবং কোথাও এটি হোস্ট করেন - না ... এটি আপাতত এসওতে এখানে বাস করে।
mattmc3

3
@ mattmc3 আপনার কোড স্যার এর জন্য আপনাকে ধন্যবাদ, তবে পাবলিক ডোমেন ইস্যুতে আপনার মন্তব্য আমাকে উদ্বেগ জানায়, যখন আপনি মন্তব্যে বলেছিলেন: "আপনি যদি এটির লাইসেন্স বোঝাতে চান এবং এটি কোথাও হোস্ট করেছেন - না ... এটি কেবলমাত্র আপাতত এখানেই বাস করে। " লেখকের প্রতি সমস্ত শ্রদ্ধার সাথে (সত্যই বোঝানো), আপনি যদি এসও-তে কোড পোস্ট করেন তবে আপনার কি সেই বিধিনিষেধ তৈরি করার অধিকার আছে ??? উদাহরণস্বরূপ, গিট গিস্ট হিসাবে, আমাদের কারও কি এসও তে আমাদের কোড পোস্ট করা থেকে সীমাবদ্ধ করার অধিকার রয়েছে? যে কেউ?
নিকোলাস পিটারসেন

6
@ নিকোলাস পিটারসন - আমার মনে হয় আপনি ভুল বুঝে গেছেন। কর্নেল প্যানিকের প্রত্যুত্তর হিসাবে আমি তাকে জানিয়েছিলাম যে আমি ব্যক্তিগতভাবে এটি লাইসেন্স করি নি বা এটি অন্য কোথাও হোস্ট করি নি। (আসলে, আপনি যেহেতু এটি উল্লেখ করেছেন, আমি একটি টুকরোটি তৈরি করেছি: gist.github.com/mattmc3/6486878 )। তবে এটি লাইসেন্স ফ্রি কোড। এটি নিন এবং এটি দিয়ে যা চান তা করুন। আমি এটি আমার নিজস্ব ব্যবহারের জন্য 100% লিখেছি। এটি নির্বিঘ্ন উপভোগ করুন। প্রকৃতপক্ষে, মাইক্রোসফ্ট থেকে কেউ যদি এটি পড়েন তবে আমি তাদের সম্পূর্ণরূপে তাদের দায়িত্ব পালন করার প্রত্যাশা করি এবং অবশেষে এটি নেট। এর পরবর্তী প্রকাশে রাখি। কোন বিশেষণ প্রয়োজন।
mattmc3

2
যদি TKeyহয় int? this[]এমন ক্ষেত্রে কীভাবে কাজ করবে?
ভিবি

2
@ ক্লিকার - কেবল নিয়মিত অ্যারে স্টাইল সূচক ব্যবহার করুন। সন্নিবেশ ক্রম ঠিক একটি তালিকার মতো বজায় রাখা হয়। প্রকার রূপান্তরটি আপনাকে নির্ধারণ করে যে আপনি কোন ইনট সহ সূচী দিয়েছিলেন, বা কী এর ধরণের মাধ্যমে রেফারেন্স দিয়েছিলেন তা নির্ধারণ করে les যদি কীটির ধরণটি কোনও int হয় তবে আপনাকে getValue () পদ্ধতিটি ব্যবহার করতে হবে।
mattmc3

32

রেকর্ডের জন্য, একটি জেনেরিক কেয়েডক্লাকশন রয়েছে যা অবজেক্টগুলিকে কোনও ইনট এবং কী দ্বারা সূচকযুক্ত করতে দেয়। কীটি অবশ্যই মানটিতে এমবেড করা উচিত।


2
এটি অর্ডারডোরিওয়ের মতো আরম্ভের ক্রম বজায় রাখে না! আমার উত্তর দেখুন।
জোয়েলফ্যান

14
এটি যুক্ত / সন্নিবেশের ক্রম বজায় রাখে।
গিলিয়াম

হ্যাঁ এটি করে .. আপনি ছেলেরা এই ধারণাটি পেলেন যে কীডক্লেশনটি আইটেমগুলি সাজায় ... আমি এইবার দ্বিতীয়বার হোঁচট
খেয়েছি

1
এটি অবশ্যই শুরুর ক্রম বজায় রাখে। সহায়ক লিঙ্ক অন্তর্ভুক্ত stackoverflow.com/a/11802824/9344 এবং geekswithblogs.net/NewThingsILearned/archive/2010/01/07/...
টেড

+1, এটি ফ্রেমওয়ার্কের সেরা সমাধানের মতো বলে মনে হচ্ছে। যদিও আপনি ব্যবহার করতে চান প্রতিটি জোড়া ধরণের জন্য বিমূর্ত শ্রেণিটি প্রয়োগ করা এক ধরণের টানুন though আপনি এটি একটি জেনেরিক বাস্তবায়নের সাথে করতে পারেন যার জন্য একটি ইন্টারফেসের প্রয়োজন, তবে তারপরে আপনি যে ধরণের স্টোর সঞ্চয় করতে সক্ষম হতে চান তার ইন্টারফেসটি প্রয়োগ করতে হবে।
ডিসিএসনন

19

এখানে একটি উদ্ভট অনুসন্ধান রয়েছে: System.Web.Extensions.dll- এ System.Web.Util নেমস্পেস

// Type: System.Web.Util.OrderedDictionary`2
// Assembly: System.Web.Extensions, Version=4.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35
// Assembly location: C:\Windows\Microsoft.NET\Framework\v4.0.30319\System.Web.Extensions.dll

namespace System.Web.Util
{
    internal class OrderedDictionary<TKey, TValue> : IDictionary<TKey, TValue>, ICollection<KeyValuePair<TKey, TValue>>, IEnumerable<KeyValuePair<TKey, TValue>>, IEnumerable

নিশ্চিত না কেন এমএস কেন এটি সিস্টেমের পরিবর্তে সেখানে রেখেছিল Col সংগ্রহগুলি G জেনেরিক প্যাকেজ, তবে আমি ধরে নিই আপনি কেবল কোডটি পেস্ট করতে পারেন এবং এটি ব্যবহার করতে পারেন (এটি অভ্যন্তরীণ, সুতরাং এটি সরাসরি ব্যবহার করতে পারবেন না)। দেখে মনে হচ্ছে বাস্তবায়নটি একটি স্ট্যান্ডার্ড অভিধান এবং পৃথক কী / মান তালিকা ব্যবহার করে। অনেকটাই অকপট...


2
System.Runtime.Collectionsএছাড়াও একটি অভ্যন্তরীণ রয়েছে OrderedDictionary<TKey, TValue>যা কেবল নন-জেনেরিক সংস্করণকে ঘিরে
রেখেছে

1
System.Web.Util.OrderedDictionary<TKey, TValue>অভ্যন্তরীণভাবে জেনেরিক প্রয়োগ করা হয় Dictionary। আশ্চর্যের তা বাস্তবায়ন না IListকিন্তুICollection<KeyValuePair<TKey, TValue>>
মিখাইল

1
@ আরবয় যেমনটি বলেছিলাম - এটি অভ্যন্তরীণ ছিল এবং জেনারিক বাস্তবায়ন মোড়ানো ছিল। তবে এটি 3+ বছর আগে ছিল ... কয়েকশ রৈখিকের নীচের আকারের List<KeyValuePair<TKey,TValue>>জন্য মেমরি অ্যাক্সেস প্যাটার্নের কারণে প্রতিযোগিতামূলক হবে বড় আকারের জন্য কেবল একই তালিকাটি + অনুসন্ধান Dictionary<TKey,int>হিসাবে ব্যবহার করুন । এএফআইকে এমন কোনও ডেটাস্ট্রাকচার নেই যা বিগোতে গতি / মেমরির ক্ষেত্রে আরও ভাল করে।
ভিবি

1
@rboy এখানে লিংক জেনেরিক এক , এটা রেফারেন্স অ জেনেরিক এক hashtable ব্যবহার করে। আমি সত্যিই বাজি ধরেছি যে জেনেরিক তালিকা / অ্যারেগুলিতে রৈখিক অনুসন্ধান ব্যবহার করা ছোট আকারের জন্য আরও দ্রুত হবে।
ভিবি

1
@ পিটারমোরটেনসেন System.Collections.Specialized.OrderedDictionaryজেনেরিক ধরণের নয়। মা দেখুন, আপনি লিঙ্ক করেছেন এমন ডক পৃষ্ঠায় কোনও কোণ বন্ধনী নেই: পি
ব্যবহারকারী 7610

17

এটির মূল্যের জন্য, আমি কীভাবে এটি সমাধান করেছি তা এখানে:

   public class PairList<TKey, TValue> : List<KeyValuePair<TKey, TValue>> {
        Dictionary<TKey, int> itsIndex = new Dictionary<TKey, int>();

        public void Add(TKey key, TValue value) {
            Add(new KeyValuePair<TKey, TValue>(key, value));
            itsIndex.Add(key, Count-1);
        }

        public TValue Get(TKey key) {
            var idx = itsIndex[key];
            return this[idx].Value;
        }
    }

এটি এভাবে শুরু করা যেতে পারে:

var pairList = new PairList<string, string>
    {
        { "pitcher", "Ken" },
        { "catcher", "Brad"},
        { "left fielder", "Stan"},
    };

এবং এটির মতো অ্যাক্সেস করা হয়েছে:

foreach (var pair in pairList)
{
    Console.WriteLine("position: {0}, player: {1}",
        pair.Key, pair.Value);
}

// Guaranteed to print in the order of initialization

3
ধন্যবাদ! আমি বুঝতে পারি নি যে সংগ্রহের সূচনাকারীরা Addপদ্ধতিগুলির জন্য কেবল বিশেষ সিনট্যাক্স ।
স্যাম

10
এটি অভিধান নয়। অভিধান ঘোরা অস্থির ইন্ডেক্স এবং কোন ডুপ্লিকেট চাবি
নওফাল

এখনো যদি আপনি কী দ্বারা প্রয়োজন হবে না ইন্ডেক্স (যা খুব কঠিন যোগ করার জন্য নয়) এবং dupliacte কী ঘটতে এই উপকারে আসে
stijn

1
কোড কল pairList.Add(new KeyValuePair<K,V>())(যেমন Listক্লাসের পদ্ধতি ) আপনার একটি সমস্যা রয়েছে । সেক্ষেত্রে itsIndexঅভিধানটি আপডেট করা হয়নি এবং এখন তালিকা এবং অভিধান সিঙ্কের বাইরে রয়েছে। List.Addকোনও new PairList.Add(KeyValuePair<K,V>)পদ্ধতি তৈরি করে পদ্ধতিটি আড়াল করতে পারত, বা উত্তরাধিকারের পরিবর্তে রচনা ব্যবহার করতে পারত এবং কেবল Listআবার সমস্ত পদ্ধতি প্রয়োগ করতে
পারত

1
@ নওফাল, আপনার উদ্বেগের সমাধান করার জন্য, একটি মাত্র চেক যোগ করতে পারে: নকল প্রতিরোধের if (itsindex.Contains(key)) return;শুরুতেAdd
জোয়েলফ্যান

14

এর জেনেরিক সংস্করণের একটি প্রধান ধারণাগত সমস্যা OrderedDictionaryহ'ল একটি ব্যবহারকারীরা OrderedDictionary<TKey,TValue>এটি ব্যবহার করে সংখ্যাসূচকভাবে একটি ব্যবহার করে int, বা একটি ব্যবহার করে অনুসন্ধান করে সূচী করতে সক্ষম হবেন বলে আশাবাদী TKey। যখন একমাত্র কী কী ছিল Object, যেমনটি নন-জেনেরিকের ক্ষেত্রে হয়েছিল , OrderedDictionaryততক্ষণ সূচককে দেওয়া আর্গুমেন্টের ধরণটি কী ধরণের ইনডেক্সিং অপারেশন করা উচিত তা পার্থক্য করতে যথেষ্ট to এটি যেমনটি রয়েছে তেমনি এটি স্পষ্ট নয় যে কোনওটির সূচক কীভাবে OrderedDictionary<int, TValue>আচরণ করবে।

যদি ক্লাসগুলির মত শ্রেণিগুলির Drawing.Pointএকটি বিধি প্রস্তাবিত এবং অনুসরণ করে যে টুকরোজ-পরিবর্তনীয় কাঠামোগুলি তাদের পরিবর্তনের বৈশিষ্ট্যগুলি সম্পত্তি হিসাবে না হিসাবে ক্ষেত্র হিসাবে প্রকাশ করতে পারে এবং সংশোধনকারী সম্পত্তি সেটার ব্যবহার থেকে বিরত থাকে this, তবে কোনও OrderedDictionary<TKey,TValue>দক্ষতার সাথে কোনও ByIndexসম্পত্তি যে কোনও Indexerকাঠামো ফেরত দেয় যা দক্ষতার সাথে প্রকাশ করতে পারে যার রেফারেন্স রাখে held অভিধান, এবং এর একটি সূচকযুক্ত সম্পত্তি ছিল যার গ্রাহক এবং সেটার কল করবে GetByIndexএবং SetByIndexএটির উপরে। সুতরাং, কেউ MyDict.ByIndex[5] += 3;অভিধানের ষষ্ঠ উপাদানটিতে 3 যুক্ত করার মতো কিছু বলতে পারে ।

দুর্ভাগ্যক্রমে, সংকলকটির পক্ষে এ জাতীয় কোনও জিনিস গ্রহণ করার জন্য, বক্সিংটি এড়ানোর মাধ্যমে ByIndexসম্পত্তিটি প্রতিবার স্ট্রাকচারের পরিবর্তে নতুন ক্লাসের উদাহরণ ফিরিয়ে দেওয়া দরকার ছিল, বক্সিং এড়িয়ে চলা সুবিধাগুলি হ্রাস করে।

ভিবি.এনইটি-তে, কোনও নামযুক্ত সূচকযুক্ত সম্পত্তি ব্যবহার করে এই সমস্যাটি ঘিরে MyDict.ByIndex[int]ফেলতে পারে (যার ফলে একজন সদস্য হওয়ার MyDictপরিবর্তে এর MyDict.ByIndexসদস্য হতে হবে MyDictযার মধ্যে একজন সূচক অন্তর্ভুক্ত থাকে), তবে সি # এই জাতীয় জিনিসগুলিকে অনুমতি দেয় না।

এটির প্রস্তাব দেওয়া এখনও সার্থক হতে পারে OrderedDictionary<TKey,TValue> where TKey:classতবে প্রথম স্থানে জেনেরিক সরবরাহের বেশিরভাগ কারণ হ'ল মান ধরণের সাথে তাদের ব্যবহারের অনুমতি দেওয়া।


ভাল বিন্দু যে intটাইপযুক্ত কীগুলি একটি চ্যালেঞ্জ উপস্থাপন করে তবে সম্পর্কিত SortedList<TKey, TValue>ধরণের উদাহরণ অনুসরণ করে এটি এড়ানো যেতে পারে : কেবল সমর্থনকারী কীগুলি [...]এবং .Values[...]সূচক দ্বারা অ্যাক্সেসের জন্য ব্যবহারের প্রয়োজন । ( SortedDictionary<TKey, TValue>, বিপরীতে, যা সূচক অ্যাক্সেসের জন্য অনুকূল নয়, এর ব্যবহার প্রয়োজন .ElementAt(...).Value)
এমকিলেমেন্ট 0

7

অনেকগুলি উদ্দেশ্যে আমি খুঁজে পেয়েছি যে এটি একটি দ্বারা পেতে পারে List<KeyValuePair<K, V>>। (যদি আপনার DictionaryO (n) কী-মান সন্ধানের চেয়ে আরও ভাল প্রয়োজন হয় তবে আপনার স্পষ্টত প্রসারিত করার প্রয়োজন হয় না)


আমি নিজেই একই সিদ্ধান্তে এসেছি!
পিটার

1
যেমনটি আমি বলেছি, "অনেক উদ্দেশ্যে for"
ডেভিড মোলস

2
Tuple<T1, T2>তাদের কী-মূল্যের সম্পর্ক না থাকলে আপনি এটিও ব্যবহার করতে পারেন।
সিডিএমকেই

1
আপনি কী দ্বারা সূচক করতে না পারলে কী মান জোড়ার কী লাভ?
ডিসিএসনন

1
@ ডিসিএস শ্যানন আপনি এখনও মানগুলির কীগুলি ম্যাপ করতে পারেন, আপনি এগুলি ও (এন) এর চেয়ে দ্রুত আর দেখতে পাচ্ছেন না (বা সদৃশ কীগুলির সাথে স্বয়ংক্রিয়ভাবে ডিল করুন)। এন এর ছোট মানগুলির জন্য এটি কখনও কখনও যথেষ্ট ভাল, বিশেষত এমন পরিস্থিতিতে যেখানে আপনি সাধারণত সমস্ত কীগুলির মাধ্যমে পুনরাবৃত্তি করেন।
ডেভিড মোলস

5

আছে SortedDictionary<TKey, TValue>। যদিও শব্দার্থগতভাবে নিকটে থাকলেও আমি দাবি করছি না যে এটি ঠিক তেমনটি নয় OrderedDictionaryকারণ। এমনকি কর্মক্ষমতা বৈশিষ্ট্য থেকে। তবে Dictionary<TKey, TValue>(এবং সেই পরিমাণে OrderedDictionaryএবং উত্তরগুলিতে প্রদত্ত বাস্তবায়ন) এর মধ্যে খুব আকর্ষণীয় এবং বেশ গুরুত্বপূর্ণ পার্থক্য এবং SortedDictionaryএটি হ'ল পরেরটি নীচে বাইনারি গাছ ব্যবহার করছে। এটি গুরুত্বপূর্ণ পার্থক্য কারণ এটি জেনেরিক ক্লাসে প্রয়োগ করা মেমরির সীমাবদ্ধতায় শ্রেণিকে অনাক্রম্য করে তোলে। OutOfMemoryExceptionsকী-ভ্যালু জোড়গুলির বড় সেট হ্যান্ডেল করার জন্য জেনেরিক ক্লাস ব্যবহার করা হয় তখন এই থ্রেডটি দেখুন ।

আউট অফ মেমরিএক্সসেপশন এড়ানোর জন্য অভিধান নির্মাণকারীর কাছে ক্ষমতা প্যারামিটারের সর্বাধিক মানটি কীভাবে নির্ধারণ করা যায়?


SortedDictionary যেগুলি যুক্ত করা হয়েছিল সেটির কী বা মানগুলি পাওয়ার কোনও উপায় আছে ? এটিই OrderedDictionaryঅনুমতি দেয়। বাছাই করা থেকে আলাদা ধারণা । (আমি অতীতে এই ভুলটি করেছি; আমি ভেবেছিলাম অর্ডারডেস্ট্রিয় নির্মাণকারীর সাথে তুলনামূলক সরবরাহ করা এটিকে বাছাই করে তুলবে, তবে তুলনার সাথে এটি যা করে তা হ'ল মূল সমতা নির্ধারণ করে; উদাহরণস্বরূপ স্ট্রিং-সংবেদনশীল তুলনামূলক স্ট্রিং-সংবেদনশীল কী দেখার জন্য অনুমতি দেয়))
সরঞ্জাম নির্মাতা স্টিভ

5

ঠিক আছে, এটি একটি দুর্ভাগ্যজনক ভুল। আমি পাইথনের অর্ডারডিক্ট মিস করছি

একটি অভিধান যা কীগুলি প্রথমে inোকানো হয়েছিল সেই ক্রমের কথা মনে রাখে। কোনও নতুন এন্ট্রি যদি বিদ্যমান প্রবেশকে ওভাররাইট করে, মূল সন্নিবেশ অবস্থানটি অপরিবর্তিত থাকে। একটি এন্ট্রি মোছা এবং পুনরায় প্রবেশকরণ এটিকে শেষের দিকে নিয়ে যাবে।

তাই আমি OrderedDictionary<K,V>সি # তে আমার নিজস্ব ক্লাস লিখেছি । এটা কিভাবে কাজ করে? এটি দুটি সংগ্রহ বজায় রাখে - একটি ভ্যানিলা আনর্ডার্ড ডিকশনারি এবং কীগুলির একটি আদেশযুক্ত তালিকা। এই সমাধানের সাহায্যে, স্ট্যান্ডার্ড ডিকশনারি ক্রিয়াকলাপগুলি তাদের দ্রুত জটিলতা বজায় রাখে এবং সূচী দ্বারা অনুসন্ধান করাও দ্রুত।

https://gist.github.com/hickford/5137384

এখানে ইন্টারফেস

/// <summary>
/// A dictionary that remembers the order that keys were first inserted. If a new entry overwrites an existing entry, the original insertion position is left unchanged. Deleting an entry and reinserting it will move it to the end.
/// </summary>
/// <typeparam name="TKey">The type of keys</typeparam>
/// <typeparam name="TValue">The type of values</typeparam>
public interface IOrderedDictionary<TKey, TValue> : IDictionary<TKey, TValue>
{
    /// <summary>
    /// The value of the element at the given index.
    /// </summary>
    TValue this[int index] { get; set; }

    /// <summary>
    /// Find the position of an element by key. Returns -1 if the dictionary does not contain an element with the given key.
    /// </summary>
    int IndexOf(TKey key);

    /// <summary>
    /// Insert an element at the given index.
    /// </summary>
    void Insert(int index, TKey key, TValue value);

    /// <summary>
    /// Remove the element at the given index.
    /// </summary>
    void RemoveAt(int index);
}

3

@ ভিবি থেকে মন্তব্য অনুসরণ করার জন্য এখানে সিস্টেমের একটি অ্যাক্সেসযোগ্য বাস্তবায়ন un রুনটাইম.কালেকশনস.অর্ডারড অভিধান <,> । আমি মূলত এটি প্রতিবিম্ব দ্বারা এটি অ্যাক্সেস করতে এবং একটি কারখানার মাধ্যমে এটি সরবরাহ করতে যাচ্ছিলাম কিন্তু এটি যে dll এ রয়েছে তা খুব একটা অ্যাক্সেসযোগ্য বলে মনে হয় না তাই আমি কেবল উত্সটি নিজের কাছে টানলাম।

একটি বিষয় লক্ষণীয় এখানে সূচক নিক্ষেপ করা হবে না KeyNotFoundException । আমি সেই কনভেনশনকে একেবারে ঘৃণা করি এবং এটিই ছিল এই বাস্তবায়নে আমি নেওয়া 1 স্বাধীনতা। যদি এটি আপনার কাছে গুরুত্বপূর্ণ হয় তবে কেবলমাত্র লাইনটি প্রতিস্থাপন করুন return default(TValue);। সি # 6 ব্যবহার করে ( ভিজ্যুয়াল স্টুডিও 2013 এর সাথে সামঞ্জস্যপূর্ণ )

/// <summary>
///     System.Collections.Specialized.OrderedDictionary is NOT generic.
///     This class is essentially a generic wrapper for OrderedDictionary.
/// </summary>
/// <remarks>
///     Indexer here will NOT throw KeyNotFoundException
/// </remarks>
public class OrderedDictionary<TKey, TValue> : IDictionary<TKey, TValue>, IDictionary
{
    private readonly OrderedDictionary _privateDictionary;

    public OrderedDictionary()
    {
        _privateDictionary = new OrderedDictionary();
    }

    public OrderedDictionary(IDictionary<TKey, TValue> dictionary)
    {
        if (dictionary == null) return;

        _privateDictionary = new OrderedDictionary();

        foreach (var pair in dictionary)
        {
            _privateDictionary.Add(pair.Key, pair.Value);
        }
    }

    public bool IsReadOnly => false;
    public int Count => _privateDictionary.Count;
    int ICollection.Count => _privateDictionary.Count;
    object ICollection.SyncRoot => ((ICollection)_privateDictionary).SyncRoot;
    bool ICollection.IsSynchronized => ((ICollection)_privateDictionary).IsSynchronized;

    bool IDictionary.IsFixedSize => ((IDictionary)_privateDictionary).IsFixedSize;
    bool IDictionary.IsReadOnly => _privateDictionary.IsReadOnly;
    ICollection IDictionary.Keys => _privateDictionary.Keys;
    ICollection IDictionary.Values => _privateDictionary.Values;

    void IDictionary.Add(object key, object value)
    {
        _privateDictionary.Add(key, value);
    }

    void IDictionary.Clear()
    {
        _privateDictionary.Clear();
    }

    bool IDictionary.Contains(object key)
    {
        return _privateDictionary.Contains(key);
    }

    IDictionaryEnumerator IDictionary.GetEnumerator()
    {
        return _privateDictionary.GetEnumerator();
    }

    void IDictionary.Remove(object key)
    {
        _privateDictionary.Remove(key);
    }

    object IDictionary.this[object key]
    {
        get { return _privateDictionary[key]; }
        set { _privateDictionary[key] = value; }
    }

    void ICollection.CopyTo(Array array, int index)
    {
        _privateDictionary.CopyTo(array, index);
    }

    public TValue this[TKey key]
    {
        get
        {
            if (key == null) throw new ArgumentNullException(nameof(key));

            if (_privateDictionary.Contains(key))
            {
                return (TValue) _privateDictionary[key];
            }

            return default(TValue);
        }
        set
        {
            if (key == null) throw new ArgumentNullException(nameof(key));

            _privateDictionary[key] = value;
        }
    }

    public ICollection<TKey> Keys
    {
        get
        {
            var keys = new List<TKey>(_privateDictionary.Count);

            keys.AddRange(_privateDictionary.Keys.Cast<TKey>());

            return keys.AsReadOnly();
        }
    }

    public ICollection<TValue> Values
    {
        get
        {
            var values = new List<TValue>(_privateDictionary.Count);

            values.AddRange(_privateDictionary.Values.Cast<TValue>());

            return values.AsReadOnly();
        }
    }

    public void Add(KeyValuePair<TKey, TValue> item)
    {
        Add(item.Key, item.Value);
    }

    public void Add(TKey key, TValue value)
    {
        if (key == null) throw new ArgumentNullException(nameof(key));

        _privateDictionary.Add(key, value);
    }

    public void Clear()
    {
        _privateDictionary.Clear();
    }

    public bool Contains(KeyValuePair<TKey, TValue> item)
    {
        if (item.Key == null || !_privateDictionary.Contains(item.Key))
        {
            return false;
        }

        return _privateDictionary[item.Key].Equals(item.Value);
    }

    public bool ContainsKey(TKey key)
    {
        if (key == null) throw new ArgumentNullException(nameof(key));

        return _privateDictionary.Contains(key);
    }

    public void CopyTo(KeyValuePair<TKey, TValue>[] array, int arrayIndex)
    {
        if (array == null) throw new ArgumentNullException(nameof(array));
        if (arrayIndex < 0) throw new ArgumentOutOfRangeException(nameof(arrayIndex));
        if (array.Rank > 1 || arrayIndex >= array.Length
                           || array.Length - arrayIndex < _privateDictionary.Count)
            throw new ArgumentException("Bad Copy ToArray", nameof(array));

        var index = arrayIndex;
        foreach (DictionaryEntry entry in _privateDictionary)
        {
            array[index] = 
                new KeyValuePair<TKey, TValue>((TKey) entry.Key, (TValue) entry.Value);
            index++;
        }
    }

    public IEnumerator<KeyValuePair<TKey, TValue>> GetEnumerator()
    {
        foreach (DictionaryEntry entry in _privateDictionary)
        {
            yield return 
                new KeyValuePair<TKey, TValue>((TKey) entry.Key, (TValue) entry.Value);
        }
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
        return GetEnumerator();
    }

    public bool Remove(KeyValuePair<TKey, TValue> item)
    {
        if (false == Contains(item)) return false;

        _privateDictionary.Remove(item.Key);

        return true;
    }

    public bool Remove(TKey key)
    {
        if (key == null) throw new ArgumentNullException(nameof(key));

        if (false == _privateDictionary.Contains(key)) return false;

        _privateDictionary.Remove(key);

        return true;
    }

    public bool TryGetValue(TKey key, out TValue value)
    {
        if (key == null) throw new ArgumentNullException(nameof(key));

        var keyExists = _privateDictionary.Contains(key);
        value = keyExists ? (TValue) _privateDictionary[key] : default(TValue);

        return keyExists;
    }
}

গিটহাবটিতে টানুন অনুরোধ / আলোচনা গৃহীত


3

নুগেটে "অফিসিয়াল" প্যাকেজ বিকল্পের সন্ধানকারীদের জন্য, জেনেরিক অর্ডারড ডিকোরিয়ানের একটি বাস্তবায়ন। নেট কোরএফএক্স ল্যাবে গৃহীত হয়েছে। যদি সবকিছু ঠিকঠাক হয় তবে প্রকারটি শেষ পর্যন্ত অনুমোদিত এবং মূল .NET CoreFX রেপোতে সংহত হবে।

এই বাস্তবায়ন বাতিল হয়ে যাওয়ার সম্ভাবনা রয়েছে।

প্রতিশ্রুতিবদ্ধ বাস্তবায়ন এখানে উল্লেখ করা যেতে পারে https://github.com/dotnet/corefxlab/blob/57be99a176421992e29009701a99a370983329a6/src/Mic Microsoft.Experimental. সংগ্রহ / মাইক্রোসফ্ট / সংগ্রহ / এক্সটেনশনস / অর্ডারডিকোরিসি।

নিশ্চিতভাবে এই ধরণের ব্যবহারের জন্য উপলভ্য নিউগেট প্যাকেজটি এখানে পাওয়া যাবে https://www.nuget.org/packages/Mic Microsoft.Experimental. Collections/1.0.6-e190117-3

অথবা আপনি ভিজ্যুয়াল স্টুডিওতে প্যাকেজটি ইনস্টল করতে পারেন। "মাইক্রোসফ্ট। এক্সপেরিমেন্টাল.কলেকশনস" প্যাকেজটির জন্য ব্রাউজ করুন এবং "অন্তর্ভুক্ত পূর্বরূপ" চেকবক্সটি নির্বাচিত হয়েছে কিনা তা নিশ্চিত করুন।

টাইপটি সরকারীভাবে উপলভ্য হলে এবং এই পোস্টটি আপডেট করবে update


আপনি কখন অনুমান করতে পারবেন এটি কখন প্রকাশিত হবে?
মুভেরিসেক 25'19

আমি এই গ্রন্থাগারের বিকাশে কোনও অংশ নিচ্ছি না তাই দুর্ভাগ্যক্রমে আমার কোনও ধারণা নেই। এটি সম্ভবত "অনুমোদিত" হয়ে উঠলে এটি একটি বিল্ট-ইন ফ্রেমওয়ার্ক সংগ্রহ হবে বলে সম্ভবত।
চার্লি

1

আমি OrderedDictionary<TKey, TValue>জেনেরিকটি চারপাশে মোড়ানো SortedList<TKey, TValue>এবং একটি প্রাইভেট যুক্ত করে প্রয়োগ করেছি Dictionary<TKey, int> _order। তারপরে আমি Comparer<TKey>_অর্ডার অভিধানের একটি রেফারেন্স পাস করে একটি অভ্যন্তরীণ বাস্তবায়ন তৈরি করেছি । তারপরে আমি অভ্যন্তরীণ সাজানো তালিকার জন্য এই তুলনামূলক ব্যবহার করব। এই শ্রেণিটি কন্সট্রাক্টরের কাছে দেওয়া উপাদানগুলির ক্রম এবং সংযোজনগুলির ক্রম রাখে।

SortedList<TKey, TValue>_আর্ডারে যুক্ত করা এবং অপসারণ করা হ'ল এই বাস্তবায়নটির প্রায় একই বড় হে বৈশিষ্ট্য রয়েছে ((1)। প্রতিটি উপাদান গ্রহণ করবে ('সংক্ষেপে সি # 4 বই অনুসারে, পৃষ্ঠা 292, টেবিল 7-1) 22 (ওভারহেড) + 4 (ইনট অর্ডার) + টি কে আকার (অতিরিক্ত 8 অনুমান) = 34 এর অতিরিক্ত মেমরি স্থান SortedList<TKey, TValue>দুটি বাইটের ওভারহেডের সাথে একত্রে মোট ওভারহেড 36 বাইট, এবং একই বইতে বলা হয়েছে যে নন-জেনেরিকের OrderedDictionaryওভারহেড 59 বাইট রয়েছে।

যদি আমি sorted=trueকনস্ট্রাক্টরের কাছে পাস করি তবে _আর্ডারটি মোটেই ব্যবহার করা হয় না, মোড়ক দেওয়ার জন্য ছোট্ট ওভারহেডের সাথে OrderedDictionary<TKey, TValue>হুবহু অর্থ হয়, যদি তা মোটেই SortedList<TKey, TValue>অর্থবহ হয়।

আমি না-এত-বড় আকারের রেফারেন্স অবজেক্টগুলিকে সংরক্ষণ করতে যাচ্ছি OrderedDictionary<TKey, TValue>, তাই আমার জন্য এই সিএ। ওভারহেড 36 বাইট সহনীয়।

মূল কোডটি নীচে রয়েছে। সম্পূর্ণ আপডেট কোডটি এই টুকরোটিতে রয়েছে

public class OrderedList<TKey, TValue> : IDictionary<TKey, TValue>, IDictionary
{
    private readonly Dictionary<TKey, int> _order;
    private readonly SortedList<TKey, TValue> _internalList;

    private readonly bool _sorted;
    private readonly OrderComparer _comparer;

    public OrderedList(IDictionary<TKey, TValue> dictionary, bool sorted = false)
    {
        _sorted = sorted;

        if (dictionary == null)
            dictionary = new Dictionary<TKey, TValue>();

        if (_sorted)
        {
            _internalList = new SortedList<TKey, TValue>(dictionary);
        }
        else
        {
            _order = new Dictionary<TKey, int>();
            _comparer = new OrderComparer(ref _order);
            _internalList = new SortedList<TKey, TValue>(_comparer);
            // Keep order of the IDictionary
            foreach (var kvp in dictionary)
            {
                Add(kvp);
            }
        }
    }

    public OrderedList(bool sorted = false)
        : this(null, sorted)
    {
    }

    private class OrderComparer : Comparer<TKey>
    {
        public Dictionary<TKey, int> Order { get; set; }

        public OrderComparer(ref Dictionary<TKey, int> order)
        {
            Order = order;
        }

        public override int Compare(TKey x, TKey y)
        {
            var xo = Order[x];
            var yo = Order[y];
            return xo.CompareTo(yo);
        }
    }

    private void ReOrder()
    {
        var i = 0;
        _order = _order.OrderBy(kvp => kvp.Value).ToDictionary(kvp => kvp.Key, kvp => i++);
        _comparer.Order = _order;
        _lastOrder = _order.Values.Max() + 1;
    }

    public void Add(TKey key, TValue value)
    {
        if (!_sorted)
        {
            _order.Add(key, _lastOrder);
            _lastOrder++;

            // Very rare event
            if (_lastOrder == int.MaxValue)
                ReOrder();
        }

        _internalList.Add(key, value);
    }

    public bool Remove(TKey key)
    {
        var result = _internalList.Remove(key);
        if (!_sorted)
            _order.Remove(key);
        return result;
    }

    // Other IDictionary<> + IDictionary members implementation wrapping around _internalList
    // ...
}

অন্তত চারটি ব্যবহারের ক্ষেত্রে আমি OrderedDictionaryসন্নিবেশ বা মুছে ফেলার বিষয়ে এরকম কিছু দেখতে পাচ্ছি : (1) কোনও দিনই মুছে ফেলা হবে না; (২) মুছে ফেলা হবে, তবে কী গুরুত্বপূর্ণ তা হল আইটেমগুলি ক্রম অনুসারে যুক্ত করা উচিত; সূচকের মাধ্যমে অ্যাক্সেসের প্রয়োজন নেই; (৩) কোনও আইটের সংখ্যাসূচক সূচিটি (বা কমপক্ষে হতে পারে) অবিচ্ছিন্ন থাকা উচিত এবং সংগ্রহের সময়কালে 2 বিলিয়নের বেশি আইটেম যুক্ত করা হবে না, সুতরাং যদি আইটেম # 7 সরানো হয়, তবে আর কখনও হবে না আইটেম # 7; (৪) কোনও আইটেমের সূচকটি বেঁচে থাকাদের ক্ষেত্রে তার র‌্যাঙ্ক হওয়া উচিত।
সুপারক্যাট

1
সিনেরিয়াস # 1 এর সমান্তরালে অ্যারে ব্যবহার করে পরিচালনা করা যায় Dictionary। দৃশ্যমান # 2 এবং # 3 প্রতিটি আইটেম যুক্ত করা হয় এবং একটি আইটেমের লিঙ্ক যুক্ত করা হয় এবং এর আগে বা পরে যুক্ত করা হয় এমন একটি সূচক বজায় রেখে পরিচালনা করা যেতে পারে। পরিস্থিতি # 4 হ'ল একমাত্র এটি মনে হয় যে এটি স্বেচ্ছাসেবী ক্রমের ক্রিয়াকলাপের জন্য ও (1) সম্পাদন করতে সক্ষম হওয়া উচিত নয়। ব্যবহারের ধরণগুলির উপর নির্ভর করে, বিভিন্ন অলস আপডেট কৌশলগুলি ব্যবহার করে # 4 সহায়তা করা যেতে পারে (একটি গাছে গণনা রাখুন এবং নোড এবং তার পিতামাতাকে আপডেট করার পরিবর্তে নোডকে অবৈধ করুন)।
সুপারক্যাট

1
অভ্যন্তরীণ সাজানো তালিকায় কাস্টম তুলনামূলক ব্যবহারের কারণে সন্নিবেশ ক্রমে উপাদান রয়েছে। এটি ধীর হতে পারে তবে গণনা সম্পর্কে আপনার মন্তব্য ভুল। গণনা সম্পর্কে পরীক্ষাগুলি দেখান ...
ভিবি

1
আপনি টোডিয়েশারের সাথে কোন লাইনের কথা বলছেন? এটি অভ্যন্তরীণ তালিকাকে প্রভাবিত করে না তবে কেবল অভিধান অর্ডার করে।
ভিবি

1
@ ভিবি ক্ষমা চেয়েছি, দুটোই মিস করেছি। যেমন এটি পরীক্ষা করে নি। তারপরে একমাত্র সমস্যাটি হবে সংযোজন সহ। দুটি অভিধানের অনুসন্ধান, পাশাপাশি দুটি সন্নিবেশ। আমি ডাউনভোট বাতিল করব।
নওফাল
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.