Можно ли использовать int для ключа в KeyedCollection

Часто мне нужен набор непоследовательных объектов с числовыми идентификаторами. Мне нравится использовать для этого KeyedCollection, но я думаю, что здесь есть серьезный недостаток. Если вы используете int в качестве ключа, вы больше не сможете получить доступ к членам коллекции по их индексу (collection [index] теперь действительно collection [key]). Достаточно ли это серьезная проблема, чтобы не использовать int в качестве ключа? Какая альтернатива была бы предпочтительнее? (может быть, int.ToString ()?)

Я делал это раньше без каких-либо серьезных проблем, но недавно я столкнулся с неприятной загвоздкой, когда сериализация XML для KeyedCollection выполняет нет, если ключ является int, из-за ошибка в .NET.

Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
11
0
1 736
4
Перейти к ответу Данный вопрос помечен как решенный

Ответы 4

Возможно, лучше всего добавить метод GetById(int) к типу коллекции. Вместо этого можно использовать Collection<T>, если вам не нужен другой ключ для доступа к содержащимся объектам:

public class FooCollection : Collection<Foo>
 { Dictionary<int,Foo> dict = new Dictionary<int,Foo>();

   public Foo GetById(int id) { return dict[id]; }

   public bool Contains(int id) { return  dict.Containskey(id);}

   protected override void InsertItem(Foo f)
    { dict[f.Id] = f;
      base.InsertItem(f);
    }

   protected override void ClearItems()
    { dict.Clear();
      base.ClearItems();
    }

   protected override void RemoveItem(int index)
    { dict.Remove(base.Items[index].Id);
      base.RemoveItem(index);
    }

   protected override void SetItem(int index, Foo item)
    { dict.Remove(base.Items[index].Id);
      dict[item.Id] = item;
      base.SetItem(index, item);
    }
 }









 }

Это лучший ответ имо

nawfal 14.04.2013 12:05

Ключ в KeyedCollection должен быть уникальным и быстро выводиться из собираемого объекта. Например, для класса человека это может быть свойство SSN или, возможно, даже объединение свойств FirstName и LastName (если известно, что результат уникален). Если идентификатор действительно является полем собираемого объекта, то он является допустимым кандидатом на ключ. Но, возможно, попробуйте преобразовать его как строку, чтобы избежать столкновения.

Простым решением могло бы быть преобразование int в другой тип, чтобы создать отдельный тип для разрешения перегрузки. Если вы используете struct, эта оболочка не имеет дополнительных накладных расходов:

struct Id {
    public int Value;

    public Id(int value) { Value = value; }

    override int GetHashCode() { return Value.GetHashCode(); }

    // … Equals method.
}

Это хорошая идея. Однако всегда неплохо сделать структуры неизменяемыми. Это позволяет избежать путаницы, подобной следующей: Id id; id.Value = 2; Id id2 = id; id2.Value = 4; // Что после этого будет id.Value? Многие ожидали, что это будет 4. (извините за форматирование; я бы использовал разрывы строк, если бы мог)

Neil 03.06.2015 18:49
Ответ принят как подходящий

По сути, вам нужно решить, будут ли пользователи класса сбиты с толку тем фактом, что они, например, не могут:

for(int i=0; i=< myCollection.Count; i++)
{
    ... myCollection[i] ...
}

хотя они, конечно, могут использовать foreach или использовать приведение:

for(int i=0; i=< myCollection.Count; i++)
{
    ... ((Collection<MyType>)myCollection)[i] ...
}

Это непростое решение, так как оно может легко привести к вирусным ошибкам. Я решил разрешить это в одном из своих приложений, где доступ пользователей класса был почти исключительно по ключу.

Я не уверен, что сделал бы это для общей библиотеки классов: в целом я бы избегал раскрытия KeyedCollection в общедоступном API: вместо этого я бы предоставил IList <T> в общедоступном API и потребителей API, которые при необходимости ключевого доступа можно определить собственную внутреннюю коллекцию KeyedCollection с помощью конструктора, который принимает IEnumerable <TItem> и заполняет им коллекцию. Это означает, что вы можете легко создать новую коллекцию KeyedCollection из списка, полученного из API.

Что касается сериализации, существует также проблема производительности что я сообщил в Microsoft Connect: KeyedCollection поддерживает внутренний словарь, а также список и сериализует оба - достаточно сериализовать список, поскольку словарь можно легко воссоздать при десериализации.

По этой причине, а также из-за ошибки XmlSerialization, я бы рекомендовал вам избегать сериализации KeyedCollection - вместо этого сериализуйте только список KeyedCollection.Items.

Мне не нравится предложение обернуть ваш ключ int в другой тип. Мне кажется неправильным добавлять сложность просто для того, чтобы тип мог использоваться как элемент в KeyedCollection. Я бы использовал строковый ключ (ToString) вместо этого - это скорее похоже на класс VB6 Collection.

FWIW, я спросил тот же вопрос некоторое время назад на форумах MSDN. Есть ответ от члена команды FxCop, но нет окончательных рекомендаций.

Я думаю, что согласен с тем, чтобы не раскрывать коллекцию KeyedCollection, которая использует int в качестве ключа в API. Это, вероятно, привело бы к ошибкам на другом конце. Проблемы с сериализацией, безусловно, являются хорошей причиной сериализации не самой коллекции, а только элементов (что я и сделал в моем случае).

Jon B 15.10.2008 19:14

Также существует подход к использованию класса, производного от KeyedCollection специально для целочисленных ключей, который включает метод GetByIndex или ItemAt, который использует внутреннюю коллекцию Items для получения соответствующего элемента. Это позволяет вам использовать существующую коллекцию KeyedCollection вместо подхода, описанного ниже, который предполагает расширение коллекции.

Thomas S. Trias 07.01.2011 05:41

Вы можете отключить словарь через threshold = -1

paparazzo 24.03.2012 04:49

Другие вопросы по теме