/******************************************************** * Dictionary.h * * * * XFX Generic Dictionary class definition file * * Copyright © XFX Team. All Rights Reserved * ********************************************************/ #ifndef _SYSTEM_COLLECTIONS_GENERIC_DICTIONARY_ #define _SYSTEM_COLLECTIONS_GENERIC_DICTIONARY_ #include #include #include "EqualityComparer.h" #include "Interfaces.h" #include "KeyValuePair.h" #include namespace System { namespace Collections { namespace Generic { // Represents a collection of keys and values. template class Dictionary : public ICollection >, virtual Object //public IDictionary, { private: template struct Entry { int hashCode; int next; UKey& key; UValue& value; }; static const int defaultCapacity = 4; int buckets[]; static const char* ComparerName; IEqualityComparer* comparer; int count; Entry* entries; int _actualSize; static const char* HashSizeName; static const char* KeyValuePairsName; int version; static const char* VersionName; void Add(const KeyValuePair& keyValuePair); bool Contains(const KeyValuePair& keyValuePair) const; void CopyTo(KeyValuePair array[], const int index) const; void EnsureCapacity(int capacity); int FindEntry(TKey key) const; bool Remove(const KeyValuePair& keyValuePair); void Initialize(const int capacity); void Insert(const TKey& key, const TValue& value, const bool add); void Resize(); public: // Represents the collection of keys in a Dictionary<,>. template class KeyCollection : public ICollection { private: Dictionary* _dictionary; public: int Count() const; KeyCollection(const Dictionary* dictionary); KeyCollection(const KeyCollection &obj); ~KeyCollection(); void Add(const UKey& item); void Clear(); bool Contains(const UKey& item) const; void CopyTo(UKey array[], const int arrayIndex) const; bool Remove(const UKey& item); }; // Represents the collection of values in a Dictionary<,>. template class ValueCollection : public ICollection { private: Dictionary *_dictionary; public: int Count() const; ValueCollection(const Dictionary* dictionary); ValueCollection(const ValueCollection &obj); void Add(const UValue& item); void Clear(); bool Contains(const UValue& item) const; void CopyTo(UValue array[], const int arrayIndex) const; bool Remove(const UValue& item); }; public: IEqualityComparer* Comparer() const; int Count() const; bool IsReadOnly() const; KeyCollection Keys() const; ValueCollection Values() const; TValue& operator[](const TKey& key); Dictionary(); Dictionary(const IDictionary* dictionary); Dictionary(const int capacity); virtual ~Dictionary(); void Add(const TKey& key, const TValue& value); void Clear(); bool ContainsKey(const TKey& key) const; bool ContainsValue(const TValue& value) const; bool Remove(const TKey& key); bool TryGetValue(const TKey& key, out TValue value) const; }; ////////////////////////////////////////////////////////////////////// template int Dictionary::Count() const { return count; } template bool Dictionary::IsReadOnly() const { return false; } template Dictionary::Dictionary() { entries = new Entry[defaultCapacity]; count = 0; } template Dictionary::Dictionary(const IDictionary* dictionary) { int itemCount = dictionary->Keys()->Count(); entries = new Entry[itemCount]; for (int i = 0; i < itemCount; i++) { // TODO: get items } } template Dictionary::Dictionary(const int capacity) { entries = new Entry[capacity]; count = 0; } template Dictionary::~Dictionary() { delete[] entries; } template void Dictionary::Add(const TKey& key, const TValue& value) { Insert(key, value, true); } template void Dictionary::Add(const KeyValuePair& keyValuePair) { Insert(keyValuePair.Key, keyValuePair.Value, true); } template bool Dictionary::Contains(const KeyValuePair& keyValuePair) const { for (int i = 0; i < count; i++) { if (entries[i].key == keyValuePair.Key && entries[i].value == keyValuePair.Value) return true; } return false; } template void Dictionary::CopyTo(KeyValuePair array[], const int index) const { return; } template void Dictionary::EnsureCapacity(int capacity) { if (_actualSize < capacity) { int num = (_actualSize == 0) ? defaultCapacity : _actualSize * 2; if (num > 0x7fefffff) { num = 0x7fefffff; } if (num < capacity) { num = capacity; } if (num != _actualSize) { if (num > 0) { Entry* destinationArray = new Entry[num]; if (count > 0) { Array::Copy(entries, 0, destinationArray, 0, count); } delete[] entries; entries = destinationArray; } else { delete[] entries; entries = new Entry[0]; } _actualSize = num; } } } template int Dictionary::FindEntry(TKey key) const { for (int i = 0; i < count; i++) { if (entries[i].key == key) return i; } return -1; } template void Dictionary::Insert(const TKey& key, const TValue& value, const bool add) { int index = FindEntry(key); if (index >=0) { sassert(add, "Attempting to add duplicate Key/Value pair to dictionary."); entries[index].value = value; return; } entries[count].key = key; entries[count].value = value; count++; } template bool Dictionary::Remove(const KeyValuePair& keyValuePair) { return false; } template template int Dictionary::KeyCollection::Count() const { return _dictionary->Count(); } template template Dictionary::KeyCollection::KeyCollection(const Dictionary* dictionary) { _dictionary = dictionary; } template template Dictionary::KeyCollection::KeyCollection(const KeyCollection &obj) { _dictionary = obj._dictionary; } template template Dictionary::KeyCollection::~KeyCollection() { delete _dictionary; } template template void Dictionary::KeyCollection::Add(const UKey& item) { sassert(false, "Adding keys directly to the Dictionary::Keycollection is not supported."); return; } template template void Dictionary::KeyCollection::Clear() { sassert(false, "Directly clearing the Dictionary::KeyCollection is not supported."); return; } template template bool Dictionary::KeyCollection::Contains(const UKey& item) const { return _dictionary.ContainsKey(item); } template template void Dictionary::KeyCollection::CopyTo(UKey array[], const int arrayIndex) const { sassert(array != NULL, String::Format("array; %s", FrameworkResources::ArgumentNull_Generic)); sassert(arrayIndex >=0, String::Format("arrayIndex; %s", FrameworkResources::ArgumentOutOfRange_NeedNonNegNum)); // TODO: implement } template template bool Dictionary::KeyCollection::Remove(const UKey& item) { sassert(false, "Removing keys directly from the Dictionary::KeyCollection is not supported."); return false; } template template Dictionary::ValueCollection::ValueCollection(const Dictionary* dictionary) { _dictionary = dictionary; } template void Dictionary::Clear() { // TODO: implement } template bool Dictionary::Remove(const TKey& key) { int index = FindEntry(key); if (index >= 0) { // TODO: implement } return false; } template bool Dictionary::TryGetValue(const TKey& key, out TValue value) const { int index = FindEntry(key); if(index >= 0) { value = entries[index].value; return true; } value = TValue(); return false; } template TValue& Dictionary::operator [](const TKey& key) { int index = FindEntry(key); sassert(index >= 0, ""); return entries[index]; } } } } #endif //_SYSTEM_COLLECTIONS_GENERIC_DICTIONARY_