MAP_S.CTT
上传用户:bangxh
上传日期:2007-01-31
资源大小:42235k
文件大小:14k
源码类别:

Windows编程

开发平台:

Visual C++

  1. /////////////////////////////////////////////////////////////////////////////
  2. // class CMapStringTo - a mapping from CStrings to 'VALUE's.
  3. // passed in parameters as ARG_TYPE
  4. //
  5. // optional definitions:
  6. //  IS_SERIAL   - enable serialization through CArchive extraction & insertion
  7. //  HAS_CREATE  - call constructors and destructors
  8. //
  9. // This is a part of the Microsoft Foundation Classes C++ library.
  10. // Copyright (C) 1994-1998 Microsoft Corporation
  11. // All rights reserved.
  12. //
  13. // This source code is only intended as a supplement to the
  14. // Microsoft Foundation Classes Reference and related
  15. // electronic documentation provided with the library.
  16. // See these sources for detailed information regarding the
  17. // Microsoft Foundation Classes product.
  18. /////////////////////////////////////////////////////////////////////////////
  19. //$DECLARE_TEMPLATE
  20. /////////////////////////////////////////////////////////////////////////////
  21. template<class VALUE, class ARG_VALUE>
  22. class CMapStringTo : public CObject
  23. {
  24. $ifdef IS_SERIAL
  25. DECLARE_SERIAL(CMapStringTo)
  26. $else
  27. DECLARE_DYNAMIC(CMapStringTo)
  28. $endif //!IS_SERIAL
  29. protected:
  30. // Association
  31. struct CAssoc
  32. {
  33. CAssoc* pNext;
  34. UINT nHashValue;  // needed for efficient iteration
  35. CString key;
  36. VALUE value;
  37. };
  38. public:
  39. // Construction
  40. CMapStringTo(int nBlockSize = 10);
  41. // Attributes
  42. // number of elements
  43. int GetCount() const;
  44. BOOL IsEmpty() const;
  45. // Lookup
  46. BOOL Lookup(LPCTSTR key, VALUE& rValue) const;
  47. BOOL LookupKey(LPCTSTR key, LPCTSTR& rKey) const;
  48. // Operations
  49. // Lookup and add if not there
  50. VALUE& operator[](LPCTSTR key);
  51. // add a new (key, value) pair
  52. void SetAt(LPCTSTR key, ARG_VALUE newValue);
  53. // removing existing (key, ?) pair
  54. BOOL RemoveKey(LPCTSTR key);
  55. void RemoveAll();
  56. // iterating all (key, value) pairs
  57. POSITION GetStartPosition() const;
  58. void GetNextAssoc(POSITION& rNextPosition, CString& rKey, VALUE& rValue) const;
  59. // advanced features for derived classes
  60. UINT GetHashTableSize() const;
  61. void InitHashTable(UINT hashSize, BOOL bAllocNow = TRUE);
  62. // Overridables: special non-virtual (see map implementation for details)
  63. // Routine used to user-provided hash keys
  64. UINT HashKey(LPCTSTR key) const;
  65. // Implementation
  66. protected:
  67. CAssoc** m_pHashTable;
  68. UINT m_nHashTableSize;
  69. int m_nCount;
  70. CAssoc* m_pFreeList;
  71. struct CPlex* m_pBlocks;
  72. int m_nBlockSize;
  73. CAssoc* NewAssoc();
  74. void FreeAssoc(CAssoc*);
  75. CAssoc* GetAssocAt(LPCTSTR, UINT&) const;
  76. public:
  77. ~CMapStringTo();
  78. $ifdef IS_SERIAL
  79. void Serialize(CArchive&);
  80. $endif //IS_SERIAL
  81. #ifdef _DEBUG
  82. void Dump(CDumpContext&) const;
  83. void AssertValid() const;
  84. #endif
  85. protected:
  86. // local typedefs for CTypedPtrMap class template
  87. typedef CString BASE_KEY;
  88. typedef LPCTSTR BASE_ARG_KEY;
  89. typedef VALUE BASE_VALUE;
  90. typedef ARG_VALUE BASE_ARG_VALUE;
  91. };
  92. //$IMPLEMENT_TEMPLATE_INLINES
  93. ////////////////////////////////////////////////////////////////////////////
  94. template<class VALUE, class ARG_VALUE>
  95. _AFXCOLL_INLINE int CMapStringTo<VALUE, ARG_VALUE>::GetCount() const
  96. { return m_nCount; }
  97. template<class VALUE, class ARG_VALUE>
  98. _AFXCOLL_INLINE BOOL CMapStringTo<VALUE, ARG_VALUE>::IsEmpty() const
  99. { return m_nCount == 0; }
  100. template<class VALUE, class ARG_VALUE>
  101. _AFXCOLL_INLINE void CMapStringTo<VALUE, ARG_VALUE>::SetAt(LPCTSTR key, ARG_VALUE newValue)
  102. { (*this)[key] = newValue; }
  103. template<class VALUE, class ARG_VALUE>
  104. _AFXCOLL_INLINE POSITION CMapStringTo<VALUE, ARG_VALUE>::GetStartPosition() const
  105. { return (m_nCount == 0) ? NULL : BEFORE_START_POSITION; }
  106. template<class VALUE, class ARG_VALUE>
  107. _AFXCOLL_INLINE UINT CMapStringTo<VALUE, ARG_VALUE>::GetHashTableSize() const
  108. { return m_nHashTableSize; }
  109. //$IMPLEMENT_TEMPLATE
  110. // This is a part of the Microsoft Foundation Classes C++ library.
  111. // Copyright (C) 1992-1997 Microsoft Corporation
  112. // All rights reserved.
  113. //
  114. // This source code is only intended as a supplement to the
  115. // Microsoft Foundation Classes Reference and related
  116. // electronic documentation provided with the library.
  117. // See these sources for detailed information regarding the
  118. // Microsoft Foundation Classes product.
  119. /////////////////////////////////////////////////////////////////////////////
  120. //
  121. // Implementation of parmeterized Map from CString to value
  122. //
  123. /////////////////////////////////////////////////////////////////////////////
  124. #include "stdafx.h"
  125. #ifdef AFX_COLL2_SEG
  126. #pragma code_seg(AFX_COLL2_SEG)
  127. #endif
  128. #ifdef _DEBUG
  129. #undef THIS_FILE
  130. static char THIS_FILE[] = __FILE__;
  131. #endif
  132. #include "elements.h"  // used for special creation
  133. #define new DEBUG_NEW
  134. /////////////////////////////////////////////////////////////////////////////
  135. template<class VALUE, class ARG_VALUE>
  136. CMapStringTo<VALUE, ARG_VALUE>::CMapStringTo(int nBlockSize)
  137. {
  138. ASSERT(nBlockSize > 0);
  139. m_pHashTable = NULL;
  140. m_nHashTableSize = 17;  // default size
  141. m_nCount = 0;
  142. m_pFreeList = NULL;
  143. m_pBlocks = NULL;
  144. m_nBlockSize = nBlockSize;
  145. }
  146. template<class VALUE, class ARG_VALUE>
  147. inline UINT CMapStringTo<VALUE, ARG_VALUE>::HashKey(LPCTSTR key) const
  148. {
  149. UINT nHash = 0;
  150. while (*key)
  151. nHash = (nHash<<5) + nHash + *key++;
  152. return nHash;
  153. }
  154. template<class VALUE, class ARG_VALUE>
  155. void CMapStringTo<VALUE, ARG_VALUE>::InitHashTable(
  156. UINT nHashSize, BOOL bAllocNow)
  157. //
  158. // Used to force allocation of a hash table or to override the default
  159. //   hash table size of (which is fairly small)
  160. {
  161. ASSERT_VALID(this);
  162. ASSERT(m_nCount == 0);
  163. ASSERT(nHashSize > 0);
  164. if (m_pHashTable != NULL)
  165. {
  166. // free hash table
  167. delete[] m_pHashTable;
  168. m_pHashTable = NULL;
  169. }
  170. if (bAllocNow)
  171. {
  172. m_pHashTable = new CAssoc* [nHashSize];
  173. memset(m_pHashTable, 0, sizeof(CAssoc*) * nHashSize);
  174. }
  175. m_nHashTableSize = nHashSize;
  176. }
  177. template<class VALUE, class ARG_VALUE>
  178. void CMapStringTo<VALUE, ARG_VALUE>::RemoveAll()
  179. {
  180. ASSERT_VALID(this);
  181. if (m_pHashTable != NULL)
  182. {
  183. // destroy elements
  184. for (UINT nHash = 0; nHash < m_nHashTableSize; nHash++)
  185. {
  186. CAssoc* pAssoc;
  187. for (pAssoc = m_pHashTable[nHash]; pAssoc != NULL;
  188.   pAssoc = pAssoc->pNext)
  189. {
  190. DestructElement(&pAssoc->key);  // free up string data
  191. $ifdef HAS_CREATE
  192. DestructElement(&pAssoc->value);
  193. $endif
  194. }
  195. }
  196. // free hash table
  197. delete [] m_pHashTable;
  198. m_pHashTable = NULL;
  199. }
  200. m_nCount = 0;
  201. m_pFreeList = NULL;
  202. m_pBlocks->FreeDataChain();
  203. m_pBlocks = NULL;
  204. }
  205. template<class VALUE, class ARG_VALUE>
  206. CMapStringTo<VALUE, ARG_VALUE>::~CMapStringTo()
  207. {
  208. RemoveAll();
  209. ASSERT(m_nCount == 0);
  210. }
  211. /////////////////////////////////////////////////////////////////////////////
  212. // Assoc helpers
  213. // same as CList implementation except we store CAssoc's not CNode's
  214. //    and CAssoc's are singly linked all the time
  215. template<class VALUE, class ARG_VALUE>
  216. CMapStringTo<VALUE, ARG_VALUE>::CAssoc*
  217. CMapStringTo<VALUE, ARG_VALUE>::NewAssoc()
  218. {
  219. if (m_pFreeList == NULL)
  220. {
  221. // add another block
  222. CPlex* newBlock = CPlex::Create(m_pBlocks, m_nBlockSize,
  223. sizeof(CMapStringTo::CAssoc));
  224. // chain them into free list
  225. CMapStringTo::CAssoc* pAssoc =
  226. (CMapStringTo::CAssoc*) newBlock->data();
  227. // free in reverse order to make it easier to debug
  228. pAssoc += m_nBlockSize - 1;
  229. for (int i = m_nBlockSize-1; i >= 0; i--, pAssoc--)
  230. {
  231. pAssoc->pNext = m_pFreeList;
  232. m_pFreeList = pAssoc;
  233. }
  234. }
  235. ASSERT(m_pFreeList != NULL);  // we must have something
  236. CMapStringTo::CAssoc* pAssoc = m_pFreeList;
  237. m_pFreeList = m_pFreeList->pNext;
  238. m_nCount++;
  239. ASSERT(m_nCount > 0);  // make sure we don't overflow
  240. memcpy(&pAssoc->key, &afxEmptyString, sizeof(CString));
  241. $ifdef HAS_CREATE
  242. ConstructElement(&pAssoc->value);
  243. $endif
  244. $ifdef USE_MEMSET
  245. memset(&pAssoc->value, 0, sizeof(VALUE));
  246. $endif
  247. $ifdef USE_ASSIGN
  248. pAssoc->value = 0;
  249. $endif
  250. return pAssoc;
  251. }
  252. template<class VALUE, class ARG_VALUE>
  253. void CMapStringTo<VALUE, ARG_VALUE>::FreeAssoc(CMapStringTo::CAssoc* pAssoc)
  254. {
  255. DestructElement(&pAssoc->key);  // free up string data
  256. $ifdef HAS_CREATE
  257. DestructElement(&pAssoc->value);
  258. $endif
  259. pAssoc->pNext = m_pFreeList;
  260. m_pFreeList = pAssoc;
  261. m_nCount--;
  262. ASSERT(m_nCount >= 0);  // make sure we don't underflow
  263. // if no more elements, cleanup completely
  264. if (m_nCount == 0)
  265. RemoveAll();
  266. }
  267. template<class VALUE, class ARG_VALUE>
  268. CMapStringTo<VALUE, ARG_VALUE>::CAssoc*
  269. CMapStringTo<VALUE, ARG_VALUE>::GetAssocAt(LPCTSTR key, UINT& nHash) const
  270. // find association (or return NULL)
  271. {
  272. nHash = HashKey(key) % m_nHashTableSize;
  273. if (m_pHashTable == NULL)
  274. return NULL;
  275. // see if it exists
  276. CAssoc* pAssoc;
  277. for (pAssoc = m_pHashTable[nHash]; pAssoc != NULL; pAssoc = pAssoc->pNext)
  278. {
  279. if (pAssoc->key == key)
  280. return pAssoc;
  281. }
  282. return NULL;
  283. }
  284. /////////////////////////////////////////////////////////////////////////////
  285. template<class VALUE, class ARG_VALUE>
  286. BOOL CMapStringTo<VALUE, ARG_VALUE>::Lookup(LPCTSTR key, VALUE& rValue) const
  287. {
  288. ASSERT_VALID(this);
  289. UINT nHash;
  290. CAssoc* pAssoc = GetAssocAt(key, nHash);
  291. if (pAssoc == NULL)
  292. return FALSE;  // not in map
  293. rValue = pAssoc->value;
  294. return TRUE;
  295. }
  296. template<class VALUE, class ARG_VALUE>
  297. BOOL CMapStringTo<VALUE, ARG_VALUE>::LookupKey(LPCTSTR key, LPCTSTR& rKey) const
  298. {
  299. ASSERT_VALID(this);
  300. UINT nHash;
  301. CAssoc* pAssoc = GetAssocAt(key, nHash);
  302. if (pAssoc == NULL)
  303. return FALSE;  // not in map
  304. rKey = pAssoc->key;
  305. return TRUE;
  306. }
  307. template<class VALUE, class ARG_VALUE>
  308. VALUE& CMapStringTo<VALUE, ARG_VALUE>::operator[](LPCTSTR key)
  309. {
  310. ASSERT_VALID(this);
  311. UINT nHash;
  312. CAssoc* pAssoc;
  313. if ((pAssoc = GetAssocAt(key, nHash)) == NULL)
  314. {
  315. if (m_pHashTable == NULL)
  316. InitHashTable(m_nHashTableSize);
  317. // it doesn't exist, add a new Association
  318. pAssoc = NewAssoc();
  319. pAssoc->nHashValue = nHash;
  320. pAssoc->key = key;
  321. // 'pAssoc->value' is a constructed object, nothing more
  322. // put into hash table
  323. pAssoc->pNext = m_pHashTable[nHash];
  324. m_pHashTable[nHash] = pAssoc;
  325. }
  326. return pAssoc->value;  // return new reference
  327. }
  328. template<class VALUE, class ARG_VALUE>
  329. BOOL CMapStringTo<VALUE, ARG_VALUE>::RemoveKey(LPCTSTR key)
  330. // remove key - return TRUE if removed
  331. {
  332. ASSERT_VALID(this);
  333. if (m_pHashTable == NULL)
  334. return FALSE;  // nothing in the table
  335. CAssoc** ppAssocPrev;
  336. ppAssocPrev = &m_pHashTable[HashKey(key) % m_nHashTableSize];
  337. CAssoc* pAssoc;
  338. for (pAssoc = *ppAssocPrev; pAssoc != NULL; pAssoc = pAssoc->pNext)
  339. {
  340. if (pAssoc->key == key)
  341. {
  342. // remove it
  343. *ppAssocPrev = pAssoc->pNext;  // remove from list
  344. FreeAssoc(pAssoc);
  345. return TRUE;
  346. }
  347. ppAssocPrev = &pAssoc->pNext;
  348. }
  349. return FALSE;  // not found
  350. }
  351. /////////////////////////////////////////////////////////////////////////////
  352. // Iterating
  353. template<class VALUE, class ARG_VALUE>
  354. void CMapStringTo<VALUE, ARG_VALUE>::GetNextAssoc(POSITION& rNextPosition,
  355. CString& rKey, VALUE& rValue) const
  356. {
  357. ASSERT_VALID(this);
  358. ASSERT(m_pHashTable != NULL);  // never call on empty map
  359. CAssoc* pAssocRet = (CAssoc*)rNextPosition;
  360. ASSERT(pAssocRet != NULL);
  361. if (pAssocRet == (CAssoc*) BEFORE_START_POSITION)
  362. {
  363. // find the first association
  364. for (UINT nBucket = 0; nBucket < m_nHashTableSize; nBucket++)
  365. if ((pAssocRet = m_pHashTable[nBucket]) != NULL)
  366. break;
  367. ASSERT(pAssocRet != NULL);  // must find something
  368. }
  369. // find next association
  370. ASSERT(AfxIsValidAddress(pAssocRet, sizeof(CAssoc)));
  371. CAssoc* pAssocNext;
  372. if ((pAssocNext = pAssocRet->pNext) == NULL)
  373. {
  374. // go to next bucket
  375. for (UINT nBucket = pAssocRet->nHashValue + 1;
  376.   nBucket < m_nHashTableSize; nBucket++)
  377. if ((pAssocNext = m_pHashTable[nBucket]) != NULL)
  378. break;
  379. }
  380. rNextPosition = (POSITION) pAssocNext;
  381. // fill in return data
  382. rKey = pAssocRet->key;
  383. rValue = pAssocRet->value;
  384. }
  385. $ifdef IS_SERIAL
  386. /////////////////////////////////////////////////////////////////////////////
  387. // Serialization
  388. template<class VALUE, class ARG_VALUE>
  389. void CMapStringTo<VALUE, ARG_VALUE>::Serialize(CArchive& ar)
  390. {
  391. ASSERT_VALID(this);
  392. CObject::Serialize(ar);
  393. if (ar.IsStoring())
  394. {
  395. ar.WriteCount(m_nCount);
  396. if (m_nCount == 0)
  397. return;  // nothing more to do
  398. ASSERT(m_pHashTable != NULL);
  399. for (UINT nHash = 0; nHash < m_nHashTableSize; nHash++)
  400. {
  401. CAssoc* pAssoc;
  402. for (pAssoc = m_pHashTable[nHash]; pAssoc != NULL;
  403.   pAssoc = pAssoc->pNext)
  404. {
  405. ar << pAssoc->key;
  406. ar << pAssoc->value;
  407. }
  408. }
  409. }
  410. else
  411. {
  412. DWORD nNewCount = ar.ReadCount();
  413. CString newKey;
  414. VALUE newValue;
  415. while (nNewCount--)
  416. {
  417. ar >> newKey;
  418. ar >> newValue;
  419. SetAt(newKey, newValue);
  420. }
  421. }
  422. }
  423. $endif //IS_SERIAL
  424. /////////////////////////////////////////////////////////////////////////////
  425. // Diagnostics
  426. #ifdef _DEBUG
  427. template<class VALUE, class ARG_VALUE>
  428. void CMapStringTo<VALUE, ARG_VALUE>::Dump(CDumpContext& dc) const
  429. {
  430. CObject::Dump(dc);
  431. dc << "with " << m_nCount << " elements";
  432. if (dc.GetDepth() > 0)
  433. {
  434. // Dump in format "[key] -> value"
  435. CString key;
  436. VALUE val;
  437. POSITION pos = GetStartPosition();
  438. while (pos != NULL)
  439. {
  440. GetNextAssoc(pos, key, val);
  441. dc << "nt[" << key << "] = " << val;
  442. }
  443. }
  444. dc << "n";
  445. }
  446. template<class VALUE, class ARG_VALUE>
  447. void CMapStringTo<VALUE, ARG_VALUE>::AssertValid() const
  448. {
  449. CObject::AssertValid();
  450. ASSERT(m_nHashTableSize > 0);
  451. ASSERT(m_nCount == 0 || m_pHashTable != NULL);
  452. // non-empty map should have hash table
  453. }
  454. #endif //_DEBUG
  455. #ifdef AFX_INIT_SEG
  456. #pragma code_seg(AFX_INIT_SEG)
  457. #endif
  458. $ifdef IS_SERIAL
  459. IMPLEMENT_SERIAL(CMapStringTo, CObject, 0)
  460. $else
  461. IMPLEMENT_DYNAMIC(CMapStringTo, CObject)
  462. $endif //!IS_SERIAL
  463. /////////////////////////////////////////////////////////////////////////////