home *** CD-ROM | disk | FTP | other *** search
- /*
- dict.prg
-
-
- Copyright (c) 1991 John F Kaster and Anton van Straaten
-
- Portions copyright (c) 1991 Nantucket Corp
-
- Bibliography:
- Compiler Design in C - Allen Holub
- Actor User's Manual - The Whitewater Group
- */
-
-
- #include "class(y).ch"
- #include "gen.ch"
-
-
- #define DEFAULT_HASH_SIZE 31
- #define MAX_BUCKETS 4096
-
-
-
- create class Dictionary
- instvar buckets
-
- method grow
- method getBucket
-
- export:
- method at = get // can't use 'at' coz it's reserved by Clipper
- method get
- method put
- method putAssoc
- method remove
- method do
- endclass
-
-
- constructor new(nHashSize)
- local i
-
- ifnil nHashSize := DEFAULT_HASH_SIZE
-
- ::buckets := array(nHashSize)
-
- for i := 1 to nHashSize
- ::buckets[i] := {}
- next
- return
-
-
-
- method function getBucket(cKey, nAssoc)
- local bucket := ::buckets[ CsyHashVal(cKey, len(::buckets)) ]
- local nPos := nAssoc := 0
-
- for nPos := 1 to len(bucket)
- if bucket[nPos]:key == cKey
- nAssoc := nPos
- exit
- end
- next
-
- return bucket
-
-
- method function get(cKey)
- local nAssoc
- local bucket := ::getBucket(cKey, @nAssoc)
- return if (nAssoc == 0, nil, bucket[nAssoc]:value)
-
-
- method procedure putAssoc(assoc)
- local nAssoc
- local bucket := ::getBucket(assoc:key, @nAssoc)
-
- if nAssoc == 0
- AAdd( bucket, assoc )
- nAssoc := len( bucket )
- else
- bucket[nAssoc] := assoc
- end
-
- if nAssoc > 3 .and. len(::buckets) < MAX_BUCKETS
- ::grow() // this bucket is big, grow dict
- end
-
- return
-
-
- method procedure put(cKey, value)
- local assoc := Association():new(cKey, value)
- ::putAssoc(assoc)
- return
-
-
- method procedure grow
- local i
- local nSize := min( len(::buckets) * 4 - 1, MAX_BUCKETS )
- local newDict := Dictionary():new(nSize)
-
- // rehash pairs into dict
- AEval( ::buckets, ;
- { |bucket| AEval( bucket, ;
- { |assoc| newDict:putAssoc( assoc ) } ) } )
-
- ::buckets := newDict:buckets
-
- return
-
-
- method procedure remove(cKey)
- local nAssoc
- local bucket := ::getBucket(cKey, @nAssoc)
-
- if nAssoc <> 0
- ADel( bucket, nAssoc )
- ASize( bucket, len(bucket) - 1 )
- end
- return
-
-
- method procedure do(block)
- local nBucket, nAssoc, bucket
- local buckets := ::buckets
- local nBuckets := len(buckets)
-
- // don't use aeval because of performance
- for nBucket := 1 to nBuckets
- bucket := buckets[nBucket]
- for nAssoc := 1 to len(bucket)
- eval(block, bucket[nAssoc], nAssoc)
- next
- next
- return
-
-
- // eof dict.prg
-