SNAP Library 2.1, Developer Reference
2013-09-25 10:47:25
SNAP, a general purpose, high performance system for analysis and manipulation of large networks
|
00001 #ifndef shash_h 00002 #define shash_h 00003 00005 // Hash-List-File 00006 // saves and loads hash tables into files and allows fast 00007 // iteration over the saved hash table file 00008 00009 template<class TKey, class TDat, class THashFunc = TDefaultHashFunc<TKey> > 00010 class TKeyDatFl { 00011 private: 00012 TInt ElemCnt; 00013 TFIn FIn; 00014 TKey Key; 00015 TDat Dat; 00016 public: 00017 TKeyDatFl(const TStr& FNm) : FIn(FNm) { ElemCnt.Load(FIn); } 00018 int Len() const { return ElemCnt; } 00019 bool Next() { if (FIn.Eof()) { return false; } 00020 Key.Load(FIn); Dat.Load(FIn); return true; } 00021 const TKey& GetKey() const { return Key; } 00022 TKey& GetKey() { return Key; } 00023 const TDat& GetDat() const { return Dat; } 00024 TDat& GetDat() { return Dat; } 00025 00026 static void Save(const TStr& OutFNm, const THash<TKey, TDat, THashFunc>& Hash) { 00027 TFOut FOut(OutFNm); Load(FOut, Hash); } 00028 static void Save(TSOut& SOut, const THash<TKey, TDat, THashFunc>& Hash) { 00029 SOut.Save(Hash.Len()); 00030 for (int k = Hash.FFirstKeyId(); Hash.FNextKeyId(k); ) { 00031 Hash.GetKey(k).Save(SOut); Hash[k].Save(SOut); } 00032 } 00033 static void LoadHash(const TStr& InFNm, THash<TKey, TDat, THashFunc>& Hash, const int& LoadN=-1) { 00034 TFIn FIn(InFNm); Load(FIn, Hash, LoadN); } 00035 static void LoadHash(TSIn& SIn, THash<TKey, TDat, THashFunc>& Hash, int LoadN=-1) { 00036 TInt ElemCnt(SIn); const int Start=clock(); 00037 if (ElemCnt < LoadN || LoadN == -1) { LoadN = ElemCnt; } 00038 printf("Loading %s: %d elements ... ", SIn.GetSNm().CStr(), LoadN); Hash.Gen(LoadN); 00039 for (int i = 0; i < LoadN; i++) { Hash.AddDat(TKey(SIn)).Load(SIn); } 00040 printf(" [%ds]\n", int((clock()-Start)/CLOCKS_PER_SEC)); 00041 } 00042 static void LoadKeyV(TSIn& SIn, TVec<TKey>& KeyV, int LoadN=-1) { 00043 TInt ElemCnt(SIn); const int Start=clock(); 00044 if (ElemCnt < LoadN || LoadN == -1) { LoadN = ElemCnt; } 00045 printf("Loading %s: %d elements ... ", SIn.GetSNm().CStr(), LoadN); KeyV.Gen(LoadN, 0); 00046 for (int i = 0; i < LoadN; i++) { KeyV.Add(TKey(SIn)); TDat D(SIn); } 00047 printf(" [%ds]\n", int((clock()-Start)/CLOCKS_PER_SEC)); 00048 } 00049 static void LoadDatV(TSIn& SIn, TVec<TDat>& DatV, int LoadN=-1) { 00050 TInt ElemCnt(SIn); const int Start=clock(); 00051 if (ElemCnt < LoadN || LoadN == -1) { LoadN = ElemCnt; } 00052 printf("Loading %s: %d elements ... ", SIn.GetSNm().CStr(), LoadN); DatV.Gen(LoadN, 0); 00053 for (int i = 0; i < LoadN; i++) { TKey(SIn); DatV.Add(TDat(SIn)); } 00054 printf(" [%ds]\n", int((clock()-Start)/CLOCKS_PER_SEC)); 00055 } 00056 }; 00057 00059 // Sparse Table Group 00060 template <class TVal, uint16 GroupSize> // GroupSize=48; == 32*x+16 00061 class TSparseGroup { 00062 private: 00063 unsigned char BitSet [(GroupSize-1)/8 + 1]; // fancy math is so we round up 00064 uint16 Buckets; // limits GroupSize to 64K 00065 TVal *Group; 00066 private: 00067 static int CharBit(const int& ValN) { return ValN >> 3; } 00068 static int ModBit(const int& ValN) { return 1 << (ValN&7); } 00069 bool BMTest(const int& ValN) const { return (BitSet[CharBit(ValN)] & ModBit(ValN)) != 0; } 00070 void BMSet(const int& ValN) { BitSet[CharBit(ValN)] |= ModBit(ValN); } 00071 void BMClear(const int& ValN) { BitSet[CharBit(ValN)] &= ~ModBit(ValN); } 00072 static int PosToOffset(const unsigned char *BitSet, int Pos); 00073 public: 00074 TSparseGroup() : Buckets(0), Group(NULL) { memset(BitSet, 0, sizeof(BitSet)); } 00075 TSparseGroup(TSIn& SIn) : Buckets(0), Group(NULL) { Load(SIn); } 00076 TSparseGroup(const TSparseGroup& SG); 00077 ~TSparseGroup() { if (Group != NULL) delete [] Group; } 00078 void Load(TSIn& SIn); 00079 void Save(TSOut& SOut) const; 00080 00081 TSparseGroup& operator = (const TSparseGroup& SG); 00082 bool operator == (const TSparseGroup& SG) const; 00083 bool operator < (const TSparseGroup& SG) const; 00084 00085 int Len() const { return Buckets; } 00086 int MxLen() const { return GroupSize; } 00087 int Reserved() const { return GroupSize; } 00088 bool Empty() const { return Buckets == 0; } 00089 void Clr(const bool& DoDel = true); 00090 int GetGroupSize() const { return GroupSize; } 00091 uint GetDiskSz() const { return sizeof(BitSet) + sizeof(uint16) + Buckets*sizeof(TVal); } 00092 00093 bool IsEmpty(const int& ValN) const { return ! BMTest(ValN); } 00094 const TVal& Offset(const int& Pos) const { return Group[Pos]; } 00095 TVal& Offset(const int& Pos) { return Group[Pos]; } 00096 int OffsetToPos(int Offset) const; 00097 int PosToOffset(int Pos) const { return PosToOffset(BitSet, Pos); } 00098 00099 const TVal& DefVal() const { static TVal DefValue = TVal(); return DefValue; } 00100 const TVal& Get(const int& ValN) const { 00101 if (BMTest(ValN)) return Group[PosToOffset(BitSet, ValN)]; else return DefVal(); } 00102 const TVal& operator [] (const int ValN) const { return Get(ValN); } 00103 00104 TVal& Set(const int& ValN, const TVal& Val); 00105 TVal& Set(const int& ValN) { 00106 if (! BMTest(ValN)) Set(ValN, DefVal()); 00107 return Group[PosToOffset(BitSet, ValN)]; 00108 } 00109 void Del(const int& ValN); 00110 }; 00111 00112 template <class TVal, uint16 GroupSize> 00113 int TSparseGroup<TVal, GroupSize>::PosToOffset(const unsigned char *BitSet, int Pos) { 00114 static const int bits_in [256] = { // # of bits set in one char 00115 0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 00116 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 00117 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 00118 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00119 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 00120 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00121 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00122 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 00123 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 00124 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00125 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00126 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 00127 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 00128 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 00129 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 00130 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8, 00131 }; 00132 // [Note: condition pos > 8 is an optimization; convince yourself we 00133 // give exactly the same result as if we had pos >= 8 here instead.] 00134 int Offset = 0; 00135 for ( ; Pos > 8; Pos -= 8 ) // bm[0..pos/8-1] 00136 Offset += bits_in[*BitSet++]; // chars we want *all* bits in 00137 return Offset + bits_in[*BitSet & ((1 << Pos)-1)]; // the char that includes pos 00138 } 00139 00140 template <class TVal, uint16 GroupSize> 00141 TSparseGroup<TVal, GroupSize>::TSparseGroup(const TSparseGroup& SG) : Buckets(SG.Buckets), Group(NULL) { 00142 memcpy(BitSet, SG.BitSet, sizeof(BitSet)); 00143 if (Buckets > 0) { 00144 Group = new TVal [Buckets]; 00145 for (int b = 0; b < Buckets; b++) { Group[b] = SG.Group[b]; } 00146 } 00147 } 00148 00149 template <class TVal, uint16 GroupSize> 00150 void TSparseGroup<TVal, GroupSize>::Load(TSIn& SIn) { 00151 SIn.LoadBf(BitSet, sizeof(BitSet)); 00152 SIn.Load(Buckets); 00153 if (Group != NULL) delete [] Group; 00154 Group = new TVal [Buckets]; 00155 for (int b = 0; b < Buckets; b++) { Group[b] = TVal(SIn); } 00156 } 00157 00158 template <class TVal, uint16 GroupSize> 00159 void TSparseGroup<TVal, GroupSize>::Save(TSOut& SOut) const { 00160 SOut.SaveBf(BitSet, sizeof(BitSet)); 00161 SOut.Save(Buckets); 00162 for (int b = 0; b < Buckets; b++) { Group[b].Save(SOut); } 00163 } 00164 00165 template <class TVal, uint16 GroupSize> 00166 TSparseGroup<TVal, GroupSize>& TSparseGroup<TVal, GroupSize>::operator = (const TSparseGroup& SG) { 00167 if (this != &SG) { 00168 if (SG.Buckets == 0 && Group != NULL) { 00169 delete [] Group; 00170 Group = 0; 00171 } else { 00172 if (Buckets != SG.Buckets) { 00173 if (Group != NULL) delete [] Group; 00174 Group = new TVal [SG.Buckets]; 00175 } 00176 for (int b = 0; b < SG.Buckets; b++) { Group[b] = SG.Group[b]; } 00177 } 00178 Buckets = SG.Buckets; 00179 memcpy(BitSet, SG.BitSet, sizeof(BitSet)); 00180 } 00181 return *this; 00182 } 00183 00184 template <class TVal, uint16 GroupSize> 00185 bool TSparseGroup<TVal, GroupSize>::operator == (const TSparseGroup& SG) const { 00186 if (Buckets == SG.Buckets && memcmp(BitSet, SG.BitSet, sizeof(BitSet)) == 0) { 00187 for (int b = 0; b < Buckets; b++) { 00188 if (Group[b] != SG.Group[b]) return false; 00189 } 00190 return true; 00191 } 00192 return false; 00193 } 00194 00195 template <class TVal, uint16 GroupSize> 00196 bool TSparseGroup<TVal, GroupSize>::operator < (const TSparseGroup& SG) const { 00197 if (Buckets < SG.Buckets) return true; 00198 if (memcmp(BitSet, SG.BitSet, sizeof(BitSet)) == -1) return true; 00199 for (int b = 0; b < Buckets; b++) { 00200 if (Group[b] < SG.Group[b]) return true; 00201 } 00202 return false; 00203 } 00204 00205 template <class TVal, uint16 GroupSize> 00206 int TSparseGroup<TVal, GroupSize>::OffsetToPos(int Offset) const { 00207 Assert(Offset < Buckets); 00208 for (int i = 0; i < sizeof(BitSet); i++) { 00209 for (int b = 0; b < 8; b++) { 00210 if (TB1Def::GetBit(b, BitSet[i])) { 00211 if (Offset == 0) return i*8 + b; 00212 Offset--; 00213 } 00214 } 00215 } 00216 Fail; 00217 return -1; 00218 } 00219 00220 template <class TVal, uint16 GroupSize> 00221 void TSparseGroup<TVal, GroupSize>::Clr(const bool& DoDel) { 00222 if (DoDel && Group != NULL) { 00223 delete [] Group; 00224 Group = 0; 00225 } 00226 memset(BitSet, 0, sizeof(BitSet)); 00227 Buckets = 0; 00228 } 00229 00230 template <class TVal, uint16 GroupSize> 00231 TVal& TSparseGroup<TVal, GroupSize>::Set(const int& ValN, const TVal& Val) { 00232 const int Offset = PosToOffset(BitSet, ValN); 00233 if (! BMTest(ValN)) { 00234 const TVal *OldGroup = Group; 00235 Group = new TVal [Buckets+1]; 00236 for (int b = 0; b < Offset; b++) Group[b] = OldGroup[b]; 00237 for (int b = Offset+1; b <= Buckets; b++) Group[b] = OldGroup[b-1]; 00238 if (OldGroup != NULL) delete [] OldGroup; 00239 Buckets++; 00240 BMSet(ValN); 00241 } 00242 Group[Offset] = Val; 00243 return Group[Offset]; 00244 } 00245 00246 template <class TVal, uint16 GroupSize> 00247 void TSparseGroup<TVal, GroupSize>::Del(const int& ValN) { 00248 if (BMTest(ValN)) { 00249 const int Offset = PosToOffset(BitSet, ValN); 00250 if (--Buckets == 0) { 00251 delete [] Group; 00252 Group = 0; 00253 } else { 00254 const TVal *OldGroup = Group; 00255 Group = new TVal [Buckets]; 00256 for (int b = 0; b < Offset; b++) Group[b] = OldGroup[b]; 00257 for (int b = Offset+1; b <= Buckets; b++) Group[b-1] = OldGroup[b]; 00258 if (OldGroup != NULL) delete [] OldGroup; 00259 } 00260 BMClear(ValN); 00261 } 00262 } 00263 00265 // Sparse Table Iterator 00266 template <class TVal, uint16 GroupSize> 00267 class TSparseTableI { 00268 private: 00269 typedef TSparseGroup<TVal, GroupSize> TValGroup; 00270 typedef typename TVec<TValGroup>::TIter TGroupVI; 00271 int CurOff; // Offset in the current group 00272 TGroupVI BegI, GroupI, EndI; 00273 public: 00274 TSparseTableI() : CurOff(0), GroupI(NULL), EndI(NULL) { } 00275 TSparseTableI(const TGroupVI& BegIter, const TGroupVI& CurIter, const TGroupVI& EndIter, 00276 const int& Offset = 0) : CurOff(Offset), BegI(BegIter), GroupI(CurIter), EndI(EndIter) { } 00277 TSparseTableI(const TSparseTableI& STI) : 00278 CurOff(STI.CurOff), BegI(STI.BegI), GroupI(STI.GroupI), EndI(STI.EndI) { } 00279 00280 TSparseTableI& operator = (const TSparseTableI& STI) { 00281 CurOff=STI.CurOff; BegI=STI.BegI; GroupI=STI.GroupI; EndI=STI.EndI; return *this; } 00282 bool operator == (const TSparseTableI& STI) const { 00283 return GroupI == STI.GroupI && CurOff == STI.CurOff; } 00284 bool operator < (const TSparseTableI& STI) const { 00285 return GroupI < STI.GroupI || (GroupI == STI.GroupI && CurOff < STI.CurOff); } 00286 TSparseTableI& operator++ (int) { 00287 if (CurOff+1 == GroupI->Len()) { CurOff = 0; 00288 if (GroupI < EndI) { GroupI++; 00289 while (GroupI < EndI && GroupI->Empty()) { GroupI++; } } 00290 } else { CurOff++; } 00291 return *this; 00292 } 00293 TSparseTableI& operator-- (int) { 00294 if (CurOff == 0) { 00295 while (GroupI >= BegI && GroupI->Empty()) { GroupI--; } 00296 if (GroupI >= BegI) CurOff = GroupI->Len()-1; 00297 } else { CurOff--; } 00298 return *this; 00299 } 00300 int GetValN() const { return int(GroupI-BegI)*GroupSize + GroupI->OffsetToPos(CurOff); } 00301 bool IsEnd() const { return GroupI==EndI; } 00302 TVal& operator*() const { return GroupI->Offset(CurOff); } 00303 TVal& operator()() const { return GroupI->Offset(CurOff); } 00304 TVal* operator->() const { return &(operator*()); } 00305 }; 00306 00308 // Sparse Table 00309 template <class TVal, uint16 GroupSize = 48> // == 32*x+16 00310 class TSparseTable { 00311 public: 00312 typedef TSparseGroup<TVal, GroupSize> TSGroup; 00313 typedef TSparseTableI<TVal, GroupSize> TIter; 00314 private: 00315 TInt MxVals, Vals; 00316 TVec<TSGroup> GroupV; 00317 private: 00318 static int GetGroups(const int& Vals) { return Vals == 0 ? 0 : ((Vals-1) / GroupSize) + 1; } 00319 int PosInGroup(const int& ValN) const { return ValN % GroupSize; } 00320 int GroupNum(const int& ValN) const { return ValN / GroupSize; } 00321 const TSGroup& GetGrp1(const int& ValN) const { return GroupV[GroupNum(ValN)]; } 00322 TSGroup& GetGrp1(const int& ValN) { return GroupV[GroupNum(ValN)]; } 00323 public: 00324 TSparseTable(const int& MaxVals = 0) : MxVals(MaxVals), 00325 Vals(0), GroupV(GetGroups(MaxVals), GetGroups(MaxVals)) { } 00326 TSparseTable(const TSparseTable& ST) : MxVals(ST.MxVals), Vals(ST.Vals), GroupV(ST.GroupV) { } 00327 TSparseTable(TSIn& SIn) : MxVals(SIn), Vals(SIn), GroupV(SIn) { } 00328 void Load(TSIn& SIn) { MxVals.Load(SIn); Vals.Load(SIn); GroupV.Load(SIn); } 00329 void Save(TSOut& SOut) const { MxVals.Save(SOut); Vals.Save(SOut); GroupV.Save(SOut); } 00330 00331 TSparseTable& operator = (const TSparseTable& ST); 00332 bool operator == (const TSparseTable& ST) const; 00333 bool operator < (const TSparseTable& ST) const; 00334 ::TSize GetMemUsed() const { return 2*sizeof(TInt)+Vals*sizeof(TVal)+GroupV.GetMemUsed(); } 00335 00336 TIter BegI() const { 00337 if (Len() > 0) { int B = 0; 00338 while (B < Groups() && GroupV[B].Empty()) { B++; } 00339 return TIter(GroupV.BegI(), GroupV.BegI()+B, GroupV.EndI()); } 00340 return TIter(GroupV.BegI(), GroupV.EndI(), GroupV.EndI()); 00341 } 00342 TIter EndI() const { return TIter(GroupV.BegI(), GroupV.EndI(), GroupV.EndI()); } 00343 TIter GetI(const int& ValN) const { Assert(! IsEmpty(ValN)); 00344 typedef typename TVec<TSGroup>::TIter TVIter; 00345 const TVIter GI = GroupV.GetI(GroupNum(ValN)); 00346 return TIter(GroupV.BegI(), GI, GroupV.EndI(), GI->PosToOffset(PosInGroup(ValN))); 00347 } 00348 00349 int Len() const { return Vals; } 00350 int Reserved() const { return MxVals; } 00351 int Groups() const { return GroupV.Len(); } 00352 bool Empty() const { return Vals == 0; } 00353 uint GetDiskSz() const { 00354 return sizeof(TInt)*4 + ((GroupSize+16)/8)*Groups() + sizeof(TVal)*Vals; } 00355 00356 void Clr(const bool& DoDel = true); 00357 void Reserve(const int NewVals) { Resize(NewVals); } 00358 void Resize(const int& NewVals); 00359 void Swap(TSparseTable& ST); 00360 00361 bool IsEmpty(const int& ValN) const { return GroupV[GroupNum(ValN)].IsEmpty(PosInGroup(ValN)); } 00362 const TVal& Get(const int& ValN) const { return GroupV[GroupNum(ValN)].Get(PosInGroup(ValN)); } 00363 TVal& Set(const int& ValN, const TVal& Val); 00364 TVal& Set(const int& ValN); 00365 void Del(const int& ValN); 00366 00367 TSGroup& GetGroup(const int& GroupN) { return GroupV[GroupN]; } 00368 const TSGroup& GetGroup(const int& GroupN) const { return GroupV[GroupN]; } 00369 }; 00370 00371 template <class TVal, uint16 GroupSize> 00372 TSparseTable<TVal, GroupSize>& TSparseTable<TVal, GroupSize>::operator = (const TSparseTable& ST) { 00373 if (this != &ST) { 00374 MxVals = ST.MxVals; 00375 Vals = ST.Vals; 00376 GroupV = ST.GroupV; 00377 } 00378 return *this; 00379 } 00380 00381 template <class TVal, uint16 GroupSize> 00382 bool TSparseTable<TVal, GroupSize>::operator == (const TSparseTable& ST) const { 00383 return Vals == ST.Vals && MxVals == ST.MxVals && GroupV == ST.GroupV; 00384 } 00385 00386 template <class TVal, uint16 GroupSize> 00387 bool TSparseTable<TVal, GroupSize>::operator < (const TSparseTable& ST) const { 00388 return Vals < ST.Vals || (Vals == ST.Vals && GroupV < ST.GroupV); 00389 } 00390 00391 template <class TVal, uint16 GroupSize> 00392 void TSparseTable<TVal, GroupSize>::Clr(const bool& DoDel) { 00393 if (! DoDel) { 00394 for (int g = 0; g < GroupV.Len(); g++) GroupV[g].Clr(false); 00395 } else { 00396 MxVals = 0; 00397 GroupV.Clr(true); 00398 } 00399 Vals = 0; 00400 } 00401 00402 template <class TVal, uint16 GroupSize> 00403 void TSparseTable<TVal, GroupSize>::Resize(const int& NewVals) { 00404 // only allow to grow 00405 if (NewVals > MxVals) { 00406 const int Groups = GetGroups(NewVals); 00407 GroupV.Reserve(Groups, Groups); 00408 MxVals = NewVals; 00409 } 00410 } 00411 00412 template <class TVal, uint16 GroupSize> 00413 void TSparseTable<TVal, GroupSize>::Swap(TSparseTable& ST) { 00414 ::Swap(MxVals, ST.MxVals); 00415 ::Swap(Vals, ST.Vals); 00416 GroupV.Swap(ST.GroupV); 00417 } 00418 00419 template <class TVal, uint16 GroupSize> 00420 TVal& TSparseTable<TVal, GroupSize>::Set(const int& ValN, const TVal& Val) { 00421 Assert(ValN < MxVals); 00422 TSGroup& Group = GetGrp1(ValN); 00423 const int OldVals = Group.Len(); 00424 TVal& ValRef = Group.Set(PosInGroup(ValN), Val); 00425 Vals += Group.Len() - OldVals; 00426 return ValRef; 00427 } 00428 00429 template <class TVal, uint16 GroupSize> 00430 TVal& TSparseTable<TVal, GroupSize>::Set(const int& ValN) { 00431 Assert(ValN < MxVals); 00432 TSGroup& Group = GetGrp1(ValN); 00433 const int OldVals = Group.Len(); 00434 TVal& ValRef = Group.Set(PosInGroup(ValN)); 00435 Vals += Group.Len() - OldVals; 00436 return ValRef; 00437 } 00438 00439 template <class TVal, uint16 GroupSize> 00440 void TSparseTable<TVal, GroupSize>::Del(const int& ValN) { 00441 Assert(ValN < MxVals); 00442 TSGroup& Group = GetGrp1(ValN); 00443 const int OldVals = Group.Len(); 00444 Group.Del(PosInGroup(ValN)); 00445 Vals += Group.Len() - OldVals; 00446 } 00447 00449 // Sparse Hash Key Dat 00450 #pragma pack(push, 1) // pack class size 00451 template <class TKey, class TDat> 00452 class TSHashKeyDat { 00453 public: 00454 TKey Key; 00455 TDat Dat; 00456 public: 00457 TSHashKeyDat() : Key(), Dat() { } 00458 TSHashKeyDat(const TKey& _Key) : Key(_Key), Dat() { } 00459 TSHashKeyDat(const TKey& _Key, const TDat& _Dat) : Key(_Key), Dat(_Dat) { } 00460 TSHashKeyDat(const TSHashKeyDat& HashKeyDat) : Key(HashKeyDat.Key), Dat(HashKeyDat.Dat) { } 00461 explicit TSHashKeyDat(TSIn& SIn) : Key(SIn), Dat(SIn) { } 00462 void Save(TSOut& SOut) const { Key.Save(SOut); Dat.Save(SOut); } 00463 TSHashKeyDat& operator = (const TSHashKeyDat& HashKeyDat) { if (this != &HashKeyDat) { 00464 Key = HashKeyDat.Key; Dat = HashKeyDat.Dat; } return *this; } 00465 bool operator == (const TSHashKeyDat& HashKeyDat) const { return Key == HashKeyDat.Key; } 00466 bool operator < (const TSHashKeyDat& HashKeyDat) const { return Key < HashKeyDat.Key; } 00467 int Hash() const { return Key.GetPrimHashCd(); } 00468 }; 00469 #pragma pack(pop) 00470 00472 // Sparse Hash Table 00473 template <class TKey, class TDat, uint16 GroupSize=48> // GroupSize= 32*x+16, for some x 00474 class TSparseHash { 00475 public: 00476 typedef TSHashKeyDat<TKey, TDat> THashKeyDat; 00477 typedef typename TSparseTable<THashKeyDat, GroupSize>::TIter TIter; 00478 typedef typename TSparseTable<THashKeyDat, GroupSize>::TSGroup TSGroup; 00479 public: 00480 static const float MxOccupancy; // = 0.7; //was 0.8 00481 static const float MnOccupancy; // = 0.4 * MxOccupancy; 00482 static const int MinBuckets; // = 32 (must be power of 2) 00483 private: 00484 void ResetThresh(); 00485 int GetMinSize(const int& CurVals, const int& WantedVals) const; 00486 void CopyFrom(const TSparseHash& HT, const int& MnWanted); 00487 void MoveFrom(TSparseHash& HT, const int& MnWanted); 00488 void ResizeDelta(const int& ValsToAdd, const int& MnWanted = 0); 00489 void FindPos(const TKey& Key, int& Pos, int& PosToIns) const; 00490 private: 00491 TInt ShrinkThresh, ExpandThresh; 00492 TSparseTable<THashKeyDat, GroupSize> Table; 00493 public: 00494 TSparseHash(const int& WantedVals = 0) : Table(GetMinSize(0, WantedVals)) { ResetThresh(); } 00495 TSparseHash(TSIn& SIn) : ShrinkThresh(SIn), ExpandThresh(SIn), Table(SIn) { } 00496 void Load(TSIn& SIn) { ShrinkThresh.Load(SIn); ExpandThresh.Load(SIn); Table.Load(SIn); } 00497 void Save(TSOut& SOut) const { ShrinkThresh.Save(SOut); ExpandThresh.Save(SOut); Table.Save(SOut); } 00498 00499 TSparseHash& operator = (const TSparseHash& SHT); 00500 bool operator == (const TSparseHash& SHT) const; 00501 bool operator < (const TSparseHash& SHT) const; 00502 ::TSize GetMemUsed() const { return 2*sizeof(TInt)+Table.GetMemUsed(); } 00503 00504 TIter BegI() const { return Table.BegI(); } 00505 TIter EndI() const { return Table.EndI(); } 00506 TIter GetI(const TKey& Key) const { Assert(IsKey(Key)); return Table.GetI(GetKeyId(Key)); } 00507 00508 bool Empty() const { return Len() == 0; } 00509 int Len() const { return Table.Len(); } 00510 int Reserved() const { return Table.Reserved(); } 00511 uint GetDiskSz() const { return 2*sizeof(TInt) + Table.GetDiskSz(); } 00512 00513 void Reserve(const int& MxVals) { if (MxVals > Len()) ResizeDelta(MxVals - Len(), 0); } 00514 void Clr(const bool& DoDel = true) { Table.Clr(DoDel); ResetThresh(); } 00515 void Swap(TSparseHash& HT); 00516 00517 int AddKey(const TKey& Key); 00518 TDat& AddDat(const TKey& Key); 00519 TDat& AddDat(const TKey& Key, const TDat& Dat); 00520 00521 const TKey& GetKey(const int& KeyId) const { return Table.Get(KeyId).Key; } 00522 int GetKeyId(const TKey& Key) const { 00523 int Pos, PosToIns; FindPos(Key, Pos, PosToIns); return Pos; } 00524 bool IsKey(const TKey& Key) const { return GetKeyId(Key) != -1; } 00525 bool IsKey(const TKey& Key, int& KeyId) const { 00526 KeyId = GetKeyId(Key); return KeyId != -1; } 00527 bool IsKeyId(const int& KeyId) const { return ! Table.IsEmpty(KeyId); } 00528 int GetRndKeyId(TRnd& Rnd = TInt::Rnd) const { Assert(Len()>0); 00529 int KeyId = Rnd.GetUniDevInt(Reserved()); 00530 while (! IsKeyId(KeyId)) { KeyId = Rnd.GetUniDevInt(Reserved()); } return KeyId; } 00531 00532 const TDat& GetDat(const TKey& Key) const; 00533 TDat& GetDat(const TKey& Key); 00534 const TDat& GetDatKeyId(const int& KeyId) const { return Table.Get(KeyId).Dat; } 00535 TDat& GetDatKeyId(const int& KeyId) { return Table.Set(KeyId).Dat; } 00536 void GetKeyDat(const int& KeyId, TKey& Key, TDat& Dat) const; 00537 bool IsKeyGetDat(const TKey& Key, TDat& Dat) const; 00538 00539 void GetKeyV(TVec<TKey>& KeyV) const; 00540 void GetDatV(TVec<TDat>& DatV) const; 00541 void GetKeyDatPrV(TVec<TPair<TKey, TDat> >& KeyDatPrV) const; 00542 void GetDatKeyPrV(TVec<TPair<TDat, TKey> >& DatKeyPrV) const; 00543 }; 00544 00545 template <class TKey, class TDat, uint16 GroupSize> 00546 const float TSparseHash<TKey, TDat, GroupSize>::MxOccupancy = 0.8f; //0.8f; 00547 00548 template <class TKey, class TDat, uint16 GroupSize> 00549 const float TSparseHash<TKey, TDat, GroupSize>::MnOccupancy = 0.4f * 0.8f; //0.8f 00550 00551 template <class TKey, class TDat, uint16 GroupSize> 00552 const int TSparseHash<TKey, TDat, GroupSize>::MinBuckets = 32; 00553 00554 template <class TKey, class TDat, uint16 GroupSize> 00555 void TSparseHash<TKey, TDat, GroupSize>::ResetThresh() { 00556 ExpandThresh = int(Table.Reserved() * MxOccupancy); 00557 ShrinkThresh = int(Table.Reserved() * MnOccupancy); 00558 } 00559 00560 template <class TKey, class TDat, uint16 GroupSize> 00561 int TSparseHash<TKey, TDat, GroupSize>::GetMinSize(const int& CurVals, const int& WantedVals) const { 00562 int Size = MinBuckets; 00563 while (Size*MxOccupancy < WantedVals || CurVals >= Size * MxOccupancy) Size *= 2; 00564 return Size; 00565 } 00566 00567 template <class TKey, class TDat, uint16 GroupSize> 00568 void TSparseHash<TKey, TDat, GroupSize>::CopyFrom(const TSparseHash& HT, const int& MnWanted) { 00569 Clr(false); 00570 const int NewSize = GetMinSize(HT.Reserved(), MnWanted); 00571 if (NewSize > Reserved()) { 00572 Table.Resize(NewSize); 00573 ResetThresh(); 00574 } 00575 const uint BuckM1 = Reserved() - 1; 00576 for (int g = 0; g < HT.Table.Groups(); g++) { 00577 const TSGroup& Group = HT.Table.GetGroup(g); 00578 for (int b = 0; b < Group.Len(); b++) { 00579 int Tries = 0; uint BuckNum; 00580 for (BuckNum = Group.Offset(b).Hash() & BuckM1; 00581 ! Table.IsEmpty(BuckNum); BuckNum = (BuckNum + Tries) & BuckM1) { 00582 Tries++; 00583 Assert(Tries < Reserved()); 00584 } 00585 Table.Set(BuckNum, Group.Offset(b)); 00586 } 00587 } 00588 } 00589 00590 template <class TKey, class TDat, uint16 GroupSize> 00591 void TSparseHash<TKey, TDat, GroupSize>::MoveFrom(TSparseHash& HT, const int& MnWanted) { 00592 Clr(false); 00593 int NewSize; 00594 if (MnWanted == 0) NewSize = HT.Reserved(); 00595 else NewSize = GetMinSize(HT.Reserved(), MnWanted); 00596 if (NewSize > Reserved()) { 00597 Table.Resize(NewSize); 00598 ResetThresh(); 00599 } 00600 const uint BuckM1 = Reserved() - 1; 00601 for (int g = 0; g < HT.Table.Groups(); g++) { 00602 TSGroup& Group = HT.Table.GetGroup(g); 00603 for (int b = 0; b < Group.Len(); b++) { 00604 int Tries = 0; uint BuckNum; 00605 for (BuckNum = Group.Offset(b).Hash() & BuckM1; 00606 ! Table.IsEmpty(BuckNum); BuckNum = (BuckNum + Tries) & BuckM1) { 00607 Tries++; 00608 Assert(Tries < Reserved()); 00609 } 00610 Assert(Table.IsEmpty(BuckNum)); 00611 Table.Set(BuckNum, Group.Offset(b)); 00612 } 00613 Group.Clr(true); // delete 00614 } 00615 } 00616 00617 template <class TKey, class TDat, uint16 GroupSize> 00618 void TSparseHash<TKey, TDat, GroupSize>::ResizeDelta(const int& ValsToAdd, const int& MnWanted) { 00619 if (Reserved() > MnWanted && Len()+ValsToAdd < ExpandThresh) { return; } 00620 const int NewSize = GetMinSize(Table.Len()+ValsToAdd, MnWanted); 00621 if (NewSize > Reserved()) { 00622 printf("***Resize SparseHash:%d->%d\n", Reserved(), NewSize); 00623 TSparseHash TmpHt(ValsToAdd+Len()); 00624 TmpHt.ResetThresh(); 00625 TmpHt.MoveFrom(*this, Len()); 00626 Swap(TmpHt); 00627 } 00628 } 00629 00630 template <class TKey, class TDat, uint16 GroupSize> 00631 void TSparseHash<TKey, TDat, GroupSize>::FindPos(const TKey& Key, int& Pos, int& PosToIns) const { 00632 const uint BuckM1 = Reserved() - 1; 00633 uint BuckNum = Key.GetPrimHashCd() & BuckM1; 00634 int Tries = 0; 00635 while (true) { 00636 if (Table.IsEmpty(BuckNum)) { 00637 Pos = -1; PosToIns = BuckNum; return; 00638 } 00639 else if (Key == Table.Get(BuckNum).Key) { 00640 Pos = BuckNum; PosToIns = -1; return; 00641 } 00642 Tries++; 00643 BuckNum = (BuckNum + Tries) & BuckM1; 00644 Assert(Tries < Reserved()); 00645 } 00646 } 00647 00648 template <class TKey, class TDat, uint16 GroupSize> 00649 TSparseHash<TKey, TDat, GroupSize>& TSparseHash<TKey, TDat, GroupSize>::operator = (const TSparseHash& SHT) { 00650 if (this != &SHT) { 00651 ShrinkThresh = SHT.ShrinkThresh; 00652 ExpandThresh = SHT.ExpandThresh; 00653 Table = SHT.Table; 00654 } 00655 return *this; 00656 } 00657 00658 template <class TKey, class TDat, uint16 GroupSize> 00659 bool TSparseHash<TKey, TDat, GroupSize>::operator == (const TSparseHash& SHT) const { 00660 return Table == SHT.Table; 00661 } 00662 00663 template <class TKey, class TDat, uint16 GroupSize> 00664 bool TSparseHash<TKey, TDat, GroupSize>::operator < (const TSparseHash& SHT) const { 00665 return Table < SHT.Table; 00666 } 00667 00668 template <class TKey, class TDat, uint16 GroupSize> 00669 void TSparseHash<TKey, TDat, GroupSize>::Swap(TSparseHash& HT) { 00670 ::Swap(ShrinkThresh, HT.ShrinkThresh); 00671 ::Swap(ExpandThresh, HT.ExpandThresh); 00672 Table.Swap(HT.Table); 00673 } 00674 00675 template <class TKey, class TDat, uint16 GroupSize> 00676 int TSparseHash<TKey, TDat, GroupSize>::AddKey(const TKey& Key) { 00677 ResizeDelta(1); 00678 int Pos, PosToIns; FindPos(Key, Pos, PosToIns); 00679 if (Pos != -1) { return Pos; } // key exists 00680 else { 00681 Table.Set(PosToIns, THashKeyDat(Key)); 00682 return PosToIns; 00683 } 00684 } 00685 00686 template <class TKey, class TDat, uint16 GroupSize> 00687 TDat& TSparseHash<TKey, TDat, GroupSize>::AddDat(const TKey& Key) { 00688 ResizeDelta(1); 00689 int Pos, PosToIns; FindPos(Key, Pos, PosToIns); 00690 if (PosToIns != -1) { 00691 return Table.Set(PosToIns, THashKeyDat(Key)).Dat; 00692 } else { return Table.Set(Pos).Dat; } 00693 } 00694 00695 template <class TKey, class TDat, uint16 GroupSize> 00696 TDat& TSparseHash<TKey, TDat, GroupSize>::AddDat(const TKey& Key, const TDat& Dat) { 00697 ResizeDelta(1); 00698 int Pos, PosToIns; FindPos(Key, Pos, PosToIns); 00699 if (PosToIns != -1) { 00700 return Table.Set(PosToIns, THashKeyDat(Key, Dat)).Dat; 00701 } else { return Table.Set(Pos).Dat = Dat; } 00702 } 00703 00704 template <class TKey, class TDat, uint16 GroupSize> 00705 const TDat& TSparseHash<TKey, TDat, GroupSize>::GetDat(const TKey& Key) const { 00706 int Pos, PosToIns; 00707 FindPos(Key, Pos, PosToIns); 00708 Assert(Pos != -1); 00709 return Table.Get(Pos).Dat; 00710 } 00711 00712 template <class TKey, class TDat, uint16 GroupSize> 00713 TDat& TSparseHash<TKey, TDat, GroupSize>::GetDat(const TKey& Key) { 00714 int Pos, PosToIns; 00715 FindPos(Key, Pos, PosToIns); 00716 Assert(Pos != -1); 00717 return Table.Set(Pos).Dat; 00718 } 00719 00720 template <class TKey, class TDat, uint16 GroupSize> 00721 void TSparseHash<TKey, TDat, GroupSize>::GetKeyDat(const int& KeyId, TKey& Key, TDat& Dat) const { 00722 Assert(IsKey(KeyId)); 00723 const THashKeyDat& KeyDat = Table.Get(KeyId); 00724 Key = KeyDat.Key; 00725 Dat = KeyDat.Dat; 00726 } 00727 00728 template <class TKey, class TDat, uint16 GroupSize> 00729 bool TSparseHash<TKey, TDat, GroupSize>::IsKeyGetDat(const TKey& Key, TDat& Dat) const { 00730 int KeyId; 00731 if (IsKey(Key, KeyId)) { 00732 Dat=Table.Get(KeyId).Dat; 00733 return true; 00734 } else { return false; } 00735 } 00736 00737 template <class TKey, class TDat, uint16 GroupSize> 00738 void TSparseHash<TKey, TDat, GroupSize>::GetKeyV(TVec<TKey>& KeyV) const { 00739 KeyV.Gen(Len(), 0); 00740 for (TIter i = BegI(); i < EndI(); i++) { 00741 KeyV.Add(i->Key); 00742 } 00743 } 00744 00745 template <class TKey, class TDat, uint16 GroupSize> 00746 void TSparseHash<TKey, TDat, GroupSize>::GetDatV(TVec<TDat>& DatV) const { 00747 DatV.Gen(Len(), 0); 00748 for (TIter i = BegI(); i < EndI(); i++) { 00749 DatV.Add(i->Dat); 00750 } 00751 } 00752 00753 template <class TKey, class TDat, uint16 GroupSize> 00754 void TSparseHash<TKey, TDat, GroupSize>::GetKeyDatPrV(TVec<TPair<TKey, TDat> >& KeyDatPrV) const { 00755 KeyDatPrV.Gen(Len(), 0); 00756 for (TIter i = BegI(); i < EndI(); i++) { 00757 KeyDatPrV.Add(TPair<TKey, TDat>(i->Key, i->Dat)); 00758 } 00759 } 00760 00761 template <class TKey, class TDat, uint16 GroupSize> 00762 void TSparseHash<TKey, TDat, GroupSize>::GetDatKeyPrV(TVec<TPair<TDat, TKey> >& DatKeyPrV) const { 00763 DatKeyPrV.Gen(Len(), 0); 00764 for (TIter i = BegI(); i < EndI(); i++) { 00765 DatKeyPrV.Add(TPair<TDat, TKey>(i->Dat, i->Key)); 00766 } 00767 } 00768 00770 // Sparse Set 00771 template <class TKey, uint16 GroupSize=48> // GroupSize= 32*x+16, for some x 00772 class TSparseSet { 00773 public: 00774 typedef typename TSparseTable<TKey, GroupSize>::TIter TIter; 00775 typedef typename TSparseTable<TKey, GroupSize>::TSGroup TSGroup; 00776 public: 00777 static const float MxOccupancy; // = 0.7; //was 0.8 00778 static const float MnOccupancy; // = 0.4 * MxOccupancy; 00779 static const int MinBuckets; // = 32 (must be power of 2) 00780 private: 00781 void ResetThresh(); 00782 int GetMinSize(const int& CurVals, const int& WantedVals) const; 00783 void CopyFrom(const TSparseSet& SSet, const int& MnWanted); 00784 void MoveFrom(TSparseSet& SSet, const int& MnWanted); 00785 void ResizeDelta(const int& ValsToAdd, const int& MnWanted = 0); 00786 void FindPos(const TKey& Key, int& Pos, int& PosToIns) const; 00787 private: 00788 TInt ShrinkThresh, ExpandThresh; 00789 TSparseTable<TKey, GroupSize> Table; 00790 public: 00791 TSparseSet(const int& WantedVals = 0) : Table(GetMinSize(0, WantedVals)) { ResetThresh(); } 00792 TSparseSet(TSIn& SIn) : ShrinkThresh(SIn), ExpandThresh(SIn), Table(SIn) { } 00793 void Load(TSIn& SIn) { ShrinkThresh.Load(SIn); ExpandThresh.Load(SIn); Table.Load(SIn); } 00794 void Save(TSOut& SOut) const { ShrinkThresh.Save(SOut); ExpandThresh.Save(SOut); Table.Save(SOut); } 00795 00796 TSparseSet& operator = (const TSparseSet& SSet); 00797 bool operator == (const TSparseSet& SSet) const; 00798 bool operator < (const TSparseSet& SSet) const; 00799 ::TSize GetMemUsed() const { return 2*sizeof(TInt)+Table.GetMemUsed(); } 00800 00801 TIter BegI() const { return Table.BegI(); } 00802 TIter EndI() const { return Table.EndI(); } 00803 TIter GetI(const int& KeyId) const { Assert(IsKeyId(KeyId)); return Table.GetI(KeyId); } 00804 00805 bool Empty() const { return Len() == 0; } 00806 int Len() const { return Table.Len(); } 00807 int Reserved() const { return Table.Reserved(); } 00808 uint GetDiskSz() const { return 2*sizeof(TInt) + Table.GetDiskSz(); } 00809 00810 void Reserve(const int& MxVals) { if (MxVals > Len()) ResizeDelta(MxVals - Len(), 0); } 00811 void Clr(const bool& DoDel = true) { Table.Clr(DoDel); ResetThresh(); } 00812 void Swap(TSparseSet& SSet); 00813 00814 int AddKey(const TKey& Key); 00815 const TKey& GetKey(const int& KeyId) const { return Table.Get(KeyId); } 00816 int GetKeyId(const TKey& Key) const { int Pos, PosToIns; 00817 FindPos(Key, Pos, PosToIns); return Pos; } 00818 bool IsKey(const TKey& Key) const { return GetKeyId(Key) != -1; } 00819 bool IsKey(const TKey& Key, int& KeyId) const { 00820 KeyId = GetKeyId(Key); return KeyId != -1; } 00821 bool IsKeyId(const int& KeyId) const { return ! Table.IsEmpty(KeyId); } 00822 int GetRndKeyId(TRnd& Rnd = TInt::Rnd) const { Assert(Len()>0); 00823 int KeyId = Rnd.GetUniDevInt(Reserved()); 00824 while (! IsKeyId(KeyId)) { KeyId = Rnd.GetUniDevInt(Reserved()); } return KeyId; } 00825 00826 void GetKeyV(TVec<TKey>& KeyV) const; 00827 }; 00828 00829 template <class TKey, uint16 GroupSize> 00830 const float TSparseSet<TKey, GroupSize>::MxOccupancy = 0.8f; 00831 00832 template <class TKey, uint16 GroupSize> 00833 const float TSparseSet<TKey, GroupSize>::MnOccupancy = 0.4f * 0.8f; 00834 00835 template <class TKey, uint16 GroupSize> 00836 const int TSparseSet<TKey, GroupSize>::MinBuckets = 32; 00837 00838 template <class TKey, uint16 GroupSize> 00839 void TSparseSet<TKey, GroupSize>::ResetThresh() { 00840 ExpandThresh = int(Table.Reserved() * MxOccupancy); 00841 ShrinkThresh = int(Table.Reserved() * MnOccupancy); 00842 } 00843 00844 template <class TKey, uint16 GroupSize> 00845 int TSparseSet<TKey, GroupSize>::GetMinSize(const int& CurVals, const int& WantedVals) const { 00846 int Size = MinBuckets; 00847 while (Size*MxOccupancy <= WantedVals || CurVals > Size * MxOccupancy) Size *= 2; 00848 return Size; 00849 } 00850 00851 template <class TKey, uint16 GroupSize> 00852 void TSparseSet<TKey, GroupSize>::CopyFrom(const TSparseSet& SSet, const int& MnWanted) { 00853 Clr(false); 00854 const int NewSize = GetMinSize(SSet.Reserved(), MnWanted); 00855 if (NewSize > Reserved()) { 00856 Table.Resize(NewSize); 00857 ResetThresh(); 00858 } 00859 const uint BuckM1 = Reserved() - 1; 00860 for (int g = 0; g < SSet.Table.Groups(); g++) { 00861 const TSGroup& Group = SSet.Table.GetGroup(g); 00862 for (int b = 0; b < Group.Len(); b++) { 00863 int Tries = 0; uint BuckNum; 00864 for (BuckNum = Group.Offset(b).GetPrimHashCd() & BuckM1; 00865 ! Table.IsEmpty(BuckNum); BuckNum = (BuckNum + Tries) & BuckM1) { 00866 Tries++; 00867 Assert(Tries < Reserved()); 00868 } 00869 Table.Set(BuckNum, Group.Offset(b)); 00870 } 00871 } 00872 } 00873 00874 template <class TKey, uint16 GroupSize> 00875 void TSparseSet<TKey, GroupSize>::MoveFrom(TSparseSet& SSet, const int& MnWanted) { 00876 Clr(false); 00877 int NewSize; 00878 if (MnWanted == 0) NewSize = SSet.Reserved(); 00879 else NewSize = GetMinSize(SSet.Reserved(), MnWanted); 00880 if (NewSize > Reserved()) { 00881 Table.Resize(NewSize); 00882 ResetThresh(); 00883 } 00884 const uint BuckM1 = Reserved() - 1; 00885 for (int g = 0; g < SSet.Table.Groups(); g++) { 00886 TSGroup& Group = SSet.Table.GetGroup(g); 00887 for (int b = 0; b < Group.Len(); b++) { 00888 int Tries = 0; uint BuckNum; 00889 for (BuckNum = Group.Offset(b).GetPrimHashCd() & BuckM1; 00890 ! Table.IsEmpty(BuckNum); BuckNum = (BuckNum + Tries) & BuckM1) { 00891 Tries++; 00892 Assert(Tries < Reserved()); 00893 } 00894 Assert(Table.IsEmpty(BuckNum)); 00895 Table.Set(BuckNum, Group.Offset(b)); 00896 } 00897 Group.Clr(true); // delete 00898 } 00899 } 00900 00901 template <class TKey, uint16 GroupSize> 00902 void TSparseSet<TKey, GroupSize>::ResizeDelta(const int& ValsToAdd, const int& MnWanted) { 00903 if (Reserved() > MnWanted && Len()+ValsToAdd < ExpandThresh) { return; } 00904 const int NewSize = GetMinSize(Table.Len()+ValsToAdd, MnWanted); 00905 if (NewSize > Reserved()) { 00906 printf("***Resize SparseSet: %d->%d\n", Reserved(), NewSize); 00907 TSparseSet TmpSSet(Len()+ValsToAdd); 00908 TmpSSet.ResetThresh(); 00909 TmpSSet.MoveFrom(*this, Len()); 00910 Swap(TmpSSet); 00911 } 00912 } 00913 00914 template <class TKey, uint16 GroupSize> 00915 void TSparseSet<TKey, GroupSize>::FindPos(const TKey& Key, int& Pos, int& PosToIns) const { 00916 const uint BuckM1 = Reserved() - 1; 00917 uint BuckNum = Key.GetPrimHashCd() & BuckM1; 00918 int Tries = 0; 00919 while (true) { 00920 if (Table.IsEmpty(BuckNum)) { 00921 Pos = -1; PosToIns = BuckNum; return; 00922 } 00923 else if (Key == Table.Get(BuckNum)) { 00924 Pos = BuckNum; PosToIns = -1; return; 00925 } 00926 Tries++; 00927 BuckNum = (BuckNum + Tries) & BuckM1; 00928 Assert(Tries < Reserved()); 00929 } 00930 } 00931 00932 template <class TKey, uint16 GroupSize> 00933 TSparseSet<TKey, GroupSize>& TSparseSet<TKey, GroupSize>::operator = (const TSparseSet& SSet) { 00934 if (this != &SSet) { 00935 ShrinkThresh = SSet.ShrinkThresh; 00936 ExpandThresh = SSet.ExpandThresh; 00937 Table = SSet.Table; 00938 } 00939 return *this; 00940 } 00941 00942 template <class TKey, uint16 GroupSize> 00943 bool TSparseSet<TKey, GroupSize>::operator == (const TSparseSet& SSet) const { 00944 return Table == SSet.Table; 00945 } 00946 00947 template <class TKey, uint16 GroupSize> 00948 bool TSparseSet<TKey, GroupSize>::operator < (const TSparseSet& SSet) const { 00949 return Table < SSet.Table; 00950 } 00951 00952 template <class TKey, uint16 GroupSize> 00953 void TSparseSet<TKey, GroupSize>::Swap(TSparseSet& SSet) { 00954 ::Swap(ShrinkThresh, SSet.ShrinkThresh); 00955 ::Swap(ExpandThresh, SSet.ExpandThresh); 00956 Table.Swap(SSet.Table); 00957 } 00958 00959 template <class TKey, uint16 GroupSize> 00960 int TSparseSet<TKey, GroupSize>::AddKey(const TKey& Key) { 00961 ResizeDelta(1); 00962 int Pos, PosToIns; FindPos(Key, Pos, PosToIns); 00963 if (Pos != -1) { return Pos; } // key exists 00964 else { 00965 Table.Set(PosToIns, Key); 00966 return PosToIns; 00967 } 00968 } 00969 00970 template <class TKey, uint16 GroupSize> 00971 void TSparseSet<TKey, GroupSize>::GetKeyV(TVec<TKey>& KeyV) const { 00972 KeyV.Gen(Len(), 0); 00973 for (TIter I = BegI(); I < EndI(); I++) { 00974 KeyV.Add(I()); } 00975 } 00976 00978 // Set-Hash-Key 00979 #pragma pack(push, 1) // pack class size 00980 template <class TKey> 00981 class THashSetKey{ 00982 public: 00983 TInt Next; 00984 TInt HashCd; 00985 TKey Key; 00986 public: 00987 THashSetKey(): 00988 Next(-1), HashCd(-1), Key() {} 00989 THashSetKey(const int& _Next, const int& _HashCd, const TKey& _Key): 00990 Next(_Next), HashCd(_HashCd), Key(_Key) {} 00991 explicit THashSetKey(TSIn& SIn): 00992 Next(SIn), HashCd(SIn), Key(SIn) {} 00993 void Save(TSOut& SOut) const { 00994 Next.Save(SOut); HashCd.Save(SOut); Key.Save(SOut); } 00995 void LoadXml(const PXmlTok& XmlTok, const TStr& Nm="") { 00996 XLoadHd(Nm); XLoad(Key); } 00997 void SaveXml(TSOut& SOut, const TStr& Nm) const { 00998 XSaveHd(Nm); XSave(Key); } 00999 01000 THashSetKey& operator=(const THashSetKey& SetKey) { 01001 if (this!=&SetKey) { Next=SetKey.Next; HashCd=SetKey.HashCd; Key=SetKey.Key; } 01002 return *this; } 01003 }; 01004 #pragma pack(pop) 01005 01007 // Set-Hash-Key-Iterator 01008 template <class TKey> 01009 class THashSetKeyI{ 01010 private: 01011 typedef THashSetKey<TKey> TSetKey; 01012 TSetKey* KeyI; 01013 TSetKey* EndI; 01014 public: 01015 THashSetKeyI(): KeyI(NULL), EndI(NULL) { } 01016 THashSetKeyI(const THashSetKeyI& _SetKeyI): 01017 KeyI(_SetKeyI.KeyI), EndI(_SetKeyI.EndI) { } 01018 THashSetKeyI(const TSetKey* _KeyI, const TSetKey* _EndI): 01019 KeyI((TSetKey*)_KeyI), EndI((TSetKey*)_EndI) { } 01020 01021 THashSetKeyI& operator=(const THashSetKeyI& SetKeyI) { 01022 KeyI=SetKeyI.KeyI; EndI=SetKeyI.EndI; return *this; } 01023 bool operator==(const THashSetKeyI& SetKeyI) const { 01024 return KeyI==SetKeyI.KeyI; } 01025 bool operator<(const THashSetKeyI& SetKeyI) const { 01026 return KeyI<SetKeyI.KeyI; } 01027 THashSetKeyI& operator++(int) { KeyI++; while (KeyI < EndI && KeyI->HashCd==-1) { KeyI++; } return *this; } 01028 THashSetKeyI& operator--(int) { do { KeyI--; } while (KeyI->HashCd==-1); return *this; } 01029 01030 const TKey& operator*() const { return KeyI->Key; } 01031 const TKey& operator()() const { return KeyI->Key; } 01032 const TKey* operator->() const { return KeyI->Key; } 01033 01034 const TKey& GetKey() const {Assert((KeyI!=NULL)&&(KeyI->HashCd!=-1)); return KeyI->Key; } 01035 }; 01036 01038 // Set-Table 01039 template <class TKey, class THashFunc = TDefaultHashFunc<TKey> > 01040 class THashSet{ 01041 public: 01042 typedef THashSetKeyI<TKey> TIter; 01043 private: 01044 typedef THashSetKey<TKey> TSetKey; 01045 TIntV PortV; 01046 TVec<TSetKey> KeyV; 01047 TBool AutoSizeP; 01048 TInt FFreeKeyId, FreeKeys; 01049 private: 01050 TSetKey& GetSetKey(const int& KeyId) { 01051 TSetKey& SetKey=KeyV[KeyId]; 01052 Assert(SetKey.HashCd!=-1); return SetKey; } 01053 const TSetKey& GetSetKey(const int& KeyId) const { 01054 const TSetKey& SetKey=KeyV[KeyId]; 01055 Assert(SetKey.HashCd!=-1); return SetKey; } 01056 uint GetNextPrime(const uint& Val) const; 01057 void Resize(); 01058 public: 01059 THashSet(): 01060 PortV(), KeyV(), 01061 AutoSizeP(true), FFreeKeyId(-1), FreeKeys(0) { } 01062 THashSet(const THashSet& Set): 01063 PortV(Set.PortV), KeyV(Set.KeyV), AutoSizeP(Set.AutoSizeP), 01064 FFreeKeyId(Set.FFreeKeyId), FreeKeys(Set.FreeKeys) { } 01065 THashSet(const int& ExpectVals, const bool& _AutoSizeP=false); 01066 explicit THashSet(const TVec<TKey>& KeyV); 01067 explicit THashSet(TSIn& SIn): 01068 PortV(SIn), KeyV(SIn), 01069 AutoSizeP(SIn), FFreeKeyId(SIn), FreeKeys(SIn) { 01070 SIn.LoadCs(); } 01071 void Load(TSIn& SIn) { 01072 PortV.Load(SIn); KeyV.Load(SIn); 01073 AutoSizeP=TBool(SIn); FFreeKeyId=TInt(SIn); FreeKeys=TInt(SIn); 01074 SIn.LoadCs(); } 01075 void Save(TSOut& SOut) const { 01076 PortV.Save(SOut); KeyV.Save(SOut); 01077 AutoSizeP.Save(SOut); FFreeKeyId.Save(SOut); FreeKeys.Save(SOut); 01078 SOut.SaveCs(); } 01079 void LoadXml(const PXmlTok& XmlTok, const TStr& Nm="") { 01080 XLoadHd(Nm); TVec<TSetKey> KeyV; XLoad(KeyV); XLoad(AutoSizeP); 01081 for (int KeyN=0; KeyN<KeyV.Len(); KeyN++) { 01082 AddKey(KeyV[KeyN].Key); }} 01083 void SaveXml(TSOut& SOut, const TStr& Nm) { 01084 Defrag(); XSaveHd(Nm); XSave(KeyV); XSave(AutoSizeP); } 01085 01086 THashSet& operator=(const THashSet& Set) { 01087 if (this!=&Set) { 01088 PortV=Set.PortV; KeyV=Set.KeyV; AutoSizeP=Set.AutoSizeP; 01089 FFreeKeyId=Set.FFreeKeyId; FreeKeys=Set.FreeKeys; } 01090 return *this; } 01091 bool operator==(const THashSet& Set) const; 01092 const TKey& operator[](const int& KeyId) const {return GetSetKey(KeyId).Key; } 01093 TKey& operator[](const int& KeyId) {return GetSetKey(KeyId).Key; } 01094 //bool operator()(const TKey& Key) {return IsKey(Key); } 01095 ::TSize GetMemUsed() const { 01096 return PortV.GetMemUsed() + KeyV.GetMemUsed() + sizeof(bool) + 2*sizeof(int); } 01097 01098 TIter BegI() const { 01099 if (Len()>0) { 01100 if (IsKeyIdEqKeyN()) { return TIter(KeyV.BegI(), KeyV.EndI()); } 01101 int FKeyId=-1; FNextKeyId(FKeyId); 01102 return TIter(KeyV.BegI()+FKeyId, KeyV.EndI()); } 01103 return TIter(KeyV.EndI(), KeyV.EndI()); 01104 } 01105 TIter EndI() const {return TIter(KeyV.EndI(), KeyV.EndI()); } 01106 TIter GetI(const TKey& Key) const {return TIter(&KeyV[GetKeyId(Key)], KeyV.EndI()); } 01107 01108 void Gen(const int& ExpectVals) { 01109 PortV.Gen(GetNextPrime(ExpectVals/2)); KeyV.Gen(ExpectVals, 0); 01110 FFreeKeyId=-1; FreeKeys=0; PortV.PutAll(TInt(-1)); } 01111 01112 void Clr(const bool& DoDel=true, const int& NoDelLim=-1); 01113 bool Empty() const {return Len()==0; } 01114 int Len() const {return KeyV.Len()-FreeKeys; } 01115 int GetPorts() const {return PortV.Len(); } 01116 bool IsAutoSize() const {return AutoSizeP; } 01117 int GetMxKeyIds() const {return KeyV.Len(); } 01118 int GetReservedKeyIds() const { return KeyV.Reserved(); } 01119 bool IsKeyIdEqKeyN() const {return FreeKeys==0; } 01120 01121 int AddKey(const TKey& Key); 01122 void AddKeyV(const TVec<TKey>& KeyV); 01123 01124 void DelKey(const TKey& Key); 01125 bool DelIfKey(const TKey& Key) { 01126 int KeyId; if (IsKey(Key, KeyId)) {DelKeyId(KeyId); return true;} return false;} 01127 void DelKeyId(const int& KeyId) {DelKey(GetKey(KeyId)); } 01128 void DelKeyIdV(const TIntV& KeyIdV) { 01129 for (int KeyIdN=0; KeyIdN<KeyIdV.Len(); KeyIdN++) {DelKeyId(KeyIdV[KeyIdN]); }} 01130 01131 void MarkDelKey(const TKey& Key); 01132 void MarkDelKeyId(const int& KeyId) {MarkDelKey(GetKey(KeyId)); } 01133 01134 const TKey& GetKey(const int& KeyId) const { 01135 return GetSetKey(KeyId).Key; } 01136 int GetKeyId(const TKey& Key) const; 01137 int GetRndKeyId(TRnd& Rnd) const { 01138 IAssert(IsKeyIdEqKeyN()); 01139 IAssert(Len()>0); 01140 return Rnd.GetUniDevInt(Len()); } 01141 bool IsKey(const TKey& Key) const {return GetKeyId(Key)!=-1; } 01142 bool IsKey(const TKey& Key, int& KeyId) const { 01143 KeyId=GetKeyId(Key); return KeyId!=-1; } 01144 bool IsKeyId(const int& KeyId) const { 01145 return (0<=KeyId)&&(KeyId<KeyV.Len())&&(KeyV[KeyId].HashCd!=-1); } 01146 01147 int FFirstKeyId() const {return 0-1; } 01148 bool FNextKeyId(int& KeyId) const; 01149 void GetKeyV(TVec<TKey>& KeyV) const; 01150 void Swap(THashSet& Set); 01151 01152 void Defrag(); 01153 void Pack() {KeyV.Pack(); } 01154 01155 static THashSet<TKey> GetSet(const TKey& Key1){ 01156 THashSet<TKey> Set(1); Set.AddKey(Key1); return Set;} 01157 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2){ 01158 THashSet<TKey> Set(2); Set.AddKey(Key1); Set.AddKey(Key2); return Set;} 01159 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3){ 01160 THashSet<TKey> Set(3); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); return Set;} 01161 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4){ 01162 THashSet<TKey> Set(4); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); return Set;} 01163 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4, const TKey& Key5){ 01164 THashSet<TKey> Set(5); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); Set.AddKey(Key5); return Set;} 01165 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4, const TKey& Key5, const TKey& Key6){ 01166 THashSet<TKey> Set(6); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); Set.AddKey(Key5); Set.AddKey(Key6); return Set;} 01167 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4, const TKey& Key5, const TKey& Key6, const TKey& Key7){ 01168 THashSet<TKey> Set(7); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); Set.AddKey(Key5); Set.AddKey(Key6); Set.AddKey(Key7); return Set;} 01169 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4, const TKey& Key5, const TKey& Key6, const TKey& Key7, const TKey& Key8){ 01170 THashSet<TKey> Set(8); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); Set.AddKey(Key5); Set.AddKey(Key6); Set.AddKey(Key7); Set.AddKey(Key8); return Set;} 01171 static THashSet<TKey> GetSet(const TKey& Key1, const TKey& Key2, const TKey& Key3, const TKey& Key4, const TKey& Key5, const TKey& Key6, const TKey& Key7, const TKey& Key8, const TKey& Key9){ 01172 THashSet<TKey> Set(9); Set.AddKey(Key1); Set.AddKey(Key2); Set.AddKey(Key3); Set.AddKey(Key4); Set.AddKey(Key5); Set.AddKey(Key6); Set.AddKey(Key7); Set.AddKey(Key8); Set.AddKey(Key9); return Set;} 01173 01174 }; 01175 01176 template <class TKey, class THashFunc> 01177 uint THashSet<TKey, THashFunc>::GetNextPrime(const uint& Val) const { 01178 uint* f=(uint*)TIntH::HashPrimeT, *m, *l=(uint*)TIntH::HashPrimeT + (int)TIntH::HashPrimes; 01179 int h, len = (int)TIntH::HashPrimes; 01180 while (len > 0) { 01181 h = len >> 1; m = f + h; 01182 if (*m < Val) { f = m; f++; len = len - h - 1; } 01183 else len = h; 01184 } 01185 return f == l ? *(l - 1) : *f; 01186 } 01187 01188 template <class TKey, class THashFunc> 01189 void THashSet<TKey, THashFunc>::Resize() { 01190 // resize & initialize port vector 01191 if (PortV.Len()==0) {PortV.Gen(17); } 01192 else if (AutoSizeP&&(KeyV.Len()>2*PortV.Len())) { 01193 PortV.Gen(GetNextPrime(PortV.Len()+1)); 01194 } else { 01195 return; 01196 } 01197 PortV.PutAll(TInt(-1)); 01198 // reSet keys 01199 for (int KeyId=0; KeyId<KeyV.Len(); KeyId++) { 01200 TSetKey& SetKey=KeyV[KeyId]; 01201 if (SetKey.HashCd!=-1) { 01202 int PortN=abs(THashFunc::GetPrimHashCd(SetKey.Key)%PortV.Len()); 01203 SetKey.Next=PortV[PortN]; 01204 PortV[PortN]=KeyId; 01205 } 01206 } 01207 } 01208 01209 template <class TKey, class THashFunc> 01210 THashSet<TKey, THashFunc>::THashSet(const int& ExpectVals, const bool& _AutoSizeP): 01211 PortV(GetNextPrime(ExpectVals/2+1)), KeyV(ExpectVals, 0), 01212 AutoSizeP(_AutoSizeP), FFreeKeyId(-1), FreeKeys(0) { 01213 PortV.PutAll(TInt(-1)); 01214 } 01215 01216 template <class TKey, class THashFunc> 01217 THashSet<TKey, THashFunc>::THashSet(const TVec<TKey>& _KeyV) : 01218 PortV(GetNextPrime(_KeyV.Len()/2+1)), KeyV(_KeyV.Len(), 0), 01219 AutoSizeP(false), FFreeKeyId(-1), FreeKeys(0) { 01220 PortV.PutAll(TInt(-1)); 01221 for (int i = 0; i < _KeyV.Len(); i++) { 01222 AddKey(_KeyV[i]); 01223 } 01224 } 01225 01226 template <class TKey, class THashFunc> 01227 bool THashSet<TKey, THashFunc>::operator==(const THashSet& Set) const { 01228 if (Len() != Set.Len()) { return false; } 01229 for (int k = FFirstKeyId(); FNextKeyId(k); k++) { 01230 if (! Set.IsKey(GetKey(k))) { return false; } 01231 } 01232 return true; 01233 } 01234 01235 template <class TKey, class THashFunc> 01236 void THashSet<TKey, THashFunc>::Clr(const bool& DoDel, const int& NoDelLim) { 01237 if (DoDel) { 01238 PortV.Clr(); KeyV.Clr(); 01239 } else { 01240 PortV.PutAll(TInt(-1)); 01241 KeyV.Clr(DoDel, NoDelLim); 01242 } 01243 FFreeKeyId=TInt(-1); FreeKeys=TInt(0); 01244 } 01245 01246 template <class TKey, class THashFunc> 01247 int THashSet<TKey, THashFunc>::AddKey(const TKey& Key) { 01248 if ((KeyV.Len()>2*PortV.Len())||PortV.Empty()) {Resize(); } 01249 int PortN=abs(THashFunc::GetPrimHashCd(Key)%PortV.Len()); 01250 int HashCd=abs(THashFunc::GetSecHashCd(Key)); 01251 int PrevKeyId=-1; 01252 int KeyId=PortV[PortN]; 01253 01254 while ((KeyId!=-1) && 01255 !((KeyV[KeyId].HashCd==HashCd) && (KeyV[KeyId].Key==Key))) { 01256 PrevKeyId=KeyId; KeyId=KeyV[KeyId].Next; } 01257 01258 if (KeyId==-1) { 01259 if (FFreeKeyId==-1) { 01260 KeyId=KeyV.Add(TSetKey(-1, HashCd, Key)); 01261 } else { 01262 KeyId=FFreeKeyId; FFreeKeyId=KeyV[FFreeKeyId].Next; FreeKeys--; 01263 KeyV[KeyId].Next = -1; 01264 KeyV[KeyId].HashCd = HashCd; 01265 KeyV[KeyId].Key = Key; 01266 } 01267 if (PrevKeyId==-1) { 01268 PortV[PortN]=KeyId; 01269 } else { 01270 KeyV[PrevKeyId].Next=KeyId; 01271 } 01272 } 01273 return KeyId; 01274 } 01275 01276 template <class TKey, class THashFunc> 01277 void THashSet<TKey, THashFunc>::AddKeyV(const TVec<TKey>& KeyV) { 01278 for (int i = 0; i < KeyV.Len(); i++) { AddKey(KeyV[i]); } 01279 } 01280 01281 template <class TKey, class THashFunc> 01282 void THashSet<TKey, THashFunc>::DelKey(const TKey& Key) { 01283 IAssert(!PortV.Empty()); 01284 int PortN=abs(THashFunc::GetPrimHashCd(Key)%PortV.Len()); 01285 int HashCd=abs(THashFunc::GetSecHashCd(Key)); 01286 int PrevKeyId=-1; 01287 int KeyId=PortV[PortN]; 01288 01289 while ((KeyId!=-1) && 01290 !((KeyV[KeyId].HashCd==HashCd) && (KeyV[KeyId].Key==Key))) { 01291 PrevKeyId=KeyId; KeyId=KeyV[KeyId].Next; } 01292 01293 IAssertR(KeyId!=-1, Key.GetStr()); 01294 if (PrevKeyId==-1) {PortV[PortN]=KeyV[KeyId].Next; } 01295 else {KeyV[PrevKeyId].Next=KeyV[KeyId].Next; } 01296 KeyV[KeyId].Next=FFreeKeyId; FFreeKeyId=KeyId; FreeKeys++; 01297 KeyV[KeyId].HashCd=TInt(-1); 01298 KeyV[KeyId].Key=TKey(); 01299 } 01300 01301 template <class TKey, class THashFunc> 01302 void THashSet<TKey, THashFunc>::MarkDelKey(const TKey& Key) { 01303 IAssert(!PortV.Empty()); 01304 int PortN=abs(THashFunc::GetPrimHashCd(Key)%PortV.Len()); 01305 int HashCd=abs(THashFunc::GetSecHashCd(Key)); 01306 int PrevKeyId=-1; 01307 int KeyId=PortV[PortN]; 01308 01309 while ((KeyId!=-1) && 01310 !((KeyV[KeyId].HashCd==HashCd) && (KeyV[KeyId].Key==Key))) { 01311 PrevKeyId=KeyId; KeyId=KeyV[KeyId].Next; } 01312 01313 IAssertR(KeyId!=-1, Key.GetStr()); 01314 if (PrevKeyId==-1) {PortV[PortN]=KeyV[KeyId].Next; } 01315 else {KeyV[PrevKeyId].Next=KeyV[KeyId].Next; } 01316 KeyV[KeyId].Next=FFreeKeyId; FFreeKeyId=KeyId; FreeKeys++; 01317 KeyV[KeyId].HashCd=TInt(-1); 01318 } 01319 01320 template <class TKey, class THashFunc> 01321 int THashSet<TKey, THashFunc>::GetKeyId(const TKey& Key) const { 01322 if (PortV.Empty()) {return -1; } 01323 int PortN=abs(THashFunc::GetPrimHashCd(Key)%PortV.Len()); 01324 int HashCd=abs(THashFunc::GetSecHashCd(Key)); 01325 int KeyId=PortV[PortN]; 01326 01327 while ((KeyId!=-1) && 01328 !((KeyV[KeyId].HashCd==HashCd) && (KeyV[KeyId].Key==Key))) { 01329 KeyId=KeyV[KeyId].Next; } 01330 return KeyId; 01331 } 01332 01333 template <class TKey, class THashFunc> 01334 bool THashSet<TKey, THashFunc>::FNextKeyId(int& KeyId) const { 01335 do {KeyId++; } while ((KeyId<KeyV.Len())&&(KeyV[KeyId].HashCd==-1)); 01336 return KeyId<KeyV.Len(); 01337 } 01338 01339 template <class TKey, class THashFunc> 01340 void THashSet<TKey, THashFunc>::GetKeyV(TVec<TKey>& KeyV) const { 01341 KeyV.Clr(); 01342 int KeyId=FFirstKeyId(); 01343 while (FNextKeyId(KeyId)) { 01344 KeyV.Add(GetKey(KeyId)); } 01345 } 01346 01347 template <class TKey, class THashFunc> 01348 void THashSet<TKey, THashFunc>::Swap(THashSet& Set) { 01349 if (this!=&Set) { 01350 PortV.Swap(Set.PortV); 01351 KeyV.Swap(Set.KeyV); 01352 ::Swap(AutoSizeP, Set.AutoSizeP); 01353 ::Swap(FFreeKeyId, Set.FFreeKeyId); 01354 ::Swap(FreeKeys, Set.FreeKeys); 01355 } 01356 } 01357 01358 template <class TKey, class THashFunc> 01359 void THashSet<TKey, THashFunc>::Defrag() { 01360 if (!IsKeyIdEqKeyN()) { 01361 THashSet<TKey> Set(PortV.Len()); 01362 int KeyId=FFirstKeyId(); 01363 while (FNextKeyId(KeyId)) { 01364 Set.AddKey(GetKey(KeyId)); 01365 } 01366 Pack(); 01367 operator=(Set); 01368 IAssert(IsKeyIdEqKeyN()); 01369 } 01370 } 01371 01373 // Common Hash Set Types 01374 typedef THashSet<TUCh> TUChSet; 01375 typedef THashSet<TInt> TIntSet; 01376 typedef THashSet<TUInt64> TUInt64Set; 01377 typedef THashSet<TFlt> TFltSet; 01378 typedef THashSet<TStr> TStrSet; 01379 typedef THashSet<TUChIntPr> TUChIntPrSet; 01380 typedef THashSet<TUChUInt64Pr> TUChUInt64PrSet; 01381 typedef THashSet<TIntPr> TIntPrSet; 01382 01384 // Packed Vec 01385 template<class TVal> 01386 class TPackVec { 01387 public: 01388 typedef TVal* TIter; 01389 private: 01390 int Vals; 01391 TVal* ValT; 01392 void ResizeDelta(const int& ValsToAdd=1); 01393 public: 01394 TPackVec() : Vals(0), ValT(NULL) { } 01395 TPackVec(const TPackVec& Vec) : Vals(0), ValT(NULL) { 01396 Gen(Vec.Len()); 01397 memcpy(ValT, Vec.ValT, sizeof(TVal)*Vals); 01398 } 01399 explicit TPackVec(const int& _Vals) : Vals(_Vals) { 01400 if (Vals==0) { ValT=NULL; } else { ValT = (TVal *) realloc(ValT, sizeof(TVal)*Vals); } } 01401 ~TPackVec() { if (ValT != NULL) { free(ValT); } } 01402 explicit TPackVec(TSIn& SIn): Vals(0), ValT(NULL) { Load(SIn); } 01403 void Load(TSIn& SIn); 01404 void Save(TSOut& SOut) const; 01405 01406 const TVal& operator [] (const int& ValN) const { return ValT[ValN]; } 01407 TVal& operator [] (const int& ValN) { return ValT[ValN]; } 01408 TPackVec<TVal>& operator = (const TPackVec<TVal>& Vec) { Gen(Vec.Len()); 01409 memcpy(ValT, Vec.ValT, sizeof(TVal)*Vals); return *this; } 01410 TVec<TVal>& operator = (const TVec<TVal>& Vec) { Gen(Vec.Len()); 01411 memcpy(ValT, Vec.ValT, sizeof(TVal)*Vals); return *this; } 01412 01413 void Gen(const int& _Vals) { 01414 if (ValT != NULL) { free(ValT); } Vals = _Vals; 01415 if (Vals==0) { ValT=NULL; } else { ValT = (TVal *) realloc(ValT, sizeof(TVal)*Vals); } } 01416 void Clr() { if (ValT != NULL) { free(ValT); ValT=NULL; } Vals = 0; } 01417 01418 bool Empty() const {return Vals==0; } 01419 int Len() const {return Vals; } 01420 const TVal& Last() const { return ValT[Len()-1]; } 01421 TVal& Last() { return ValT[Len()-1]; } 01422 01423 TIter BegI() const {return ValT; } 01424 TIter EndI() const {return ValT+Vals; } 01425 TIter GetI(const int& ValN) const { return ValT+ValN; } 01426 01427 void Add(const TVal& Val) { ResizeDelta(1); ValT[Vals-1]=Val; } 01428 void AddV(const TPackVec<TVal>& ValV) { ResizeDelta(ValV.Len()); 01429 memcpy(ValT+Vals-ValV.Len(), ValV.BegI(), sizeof(TVal)*ValV.Len()); } 01430 void AddV(const TVec<TVal>& ValV) { ResizeDelta(ValV.Len()); 01431 memcpy(ValT+Vals-ValV.Len(), ValV.BegI(), sizeof(TVal)*ValV.Len()); } 01432 void AddV(TSIn& FIn) { int NVals; FIn.Load(NVals); 01433 ResizeDelta(NVals); FIn.LoadBf(ValT+Vals-NVals, sizeof(TVal)*NVals); } 01434 01435 void Sort(const bool& Asc=true) { 01436 if (Asc) { TVec<TVal>::QSortCmp(BegI(), EndI(), TLss<TVal>()); } 01437 else { TVec<TVal>::QSortCmp(BegI(), EndI(), TGtr<TVal>()); } 01438 } 01439 }; 01440 01441 template<class TVal> 01442 void TPackVec<TVal>::ResizeDelta(const int& ValsToAdd) { 01443 if (ValsToAdd == 0) return; 01444 Vals += ValsToAdd; 01445 ValT = (TVal *) realloc(ValT, sizeof(TVal)*Vals); 01446 EAssert(ValT != NULL); 01447 } 01448 01449 template<class TVal> 01450 void TPackVec<TVal>::Load(TSIn& SIn) { 01451 SIn.Load(Vals); 01452 if (Vals==0) { 01453 if (ValT != NULL) { free(ValT); } 01454 ValT = NULL; } 01455 else { 01456 ValT = (TVal *) realloc(ValT, sizeof(TVal)*Vals); 01457 } 01458 SIn.LoadBf(ValT, sizeof(TVal)*Vals); 01459 } 01460 01461 template<class TVal> 01462 void TPackVec<TVal>::Save(TSOut& SOut) const { 01463 SOut.Save(Vals); 01464 if (Vals != 0) { 01465 SOut.SaveBf(ValT, sizeof(TVal)*Vals); } 01466 } 01467 01468 #endif