/************************************************************************* * * $RCSfile: hash.cxx,v $ * * $Revision: 1.2 $ * * last change: $Author: mhu $ $Date: 2002-11-29 17:47:06 $ * * The Contents of this file are made available subject to the terms of * either of the following licenses * * - GNU Lesser General Public License Version 2.1 * - Sun Industry Standards Source License Version 1.1 * * Sun Microsystems Inc., October, 2000 * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2000 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * * * Sun Industry Standards Source License Version 1.1 * ================================================= * The contents of this file are subject to the Sun Industry Standards * Source License Version 1.1 (the "License"); You may not use this file * except in compliance with the License. You may obtain a copy of the * License at http://www.openoffice.org/license.html. * * Software provided under this License is provided on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, * WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS, * MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING. * See the License for the specific provisions governing your rights and * obligations concerning the Software. * * The Initial Developer of the Original Code is: Sun Microsystems, Inc. * * Copyright: 2000 by Sun Microsystems, Inc. * * All Rights Reserved. * * Contributor(s): _______________________________________ * * ************************************************************************/ /****************** I N C L U D E S **************************************/ // C and C++ Includes. #include #include #include // Programmabh„ngige Includes. #include #ifndef _TOOLS_DEBUG_HXX //autogen #include #endif #pragma hdrstop /****************** C O D E **********************************************/ /************************************************************************* |* |* SvStringHashEntry::~SvStringHashEntry() |* |* Beschreibung |* *************************************************************************/ SvStringHashEntry::~SvStringHashEntry() { }; /************************************************************************* |* |* SvHashTable::SvHashTable() |* |* Beschreibung |* *************************************************************************/ SvHashTable::SvHashTable( UINT32 nMaxEntries ) { nMax = nMaxEntries; // set max entries nFill = 0; // no entries lTry = 0; lAsk = 0; } /************************************************************************* |* |* SvHashTable::~SvHashTable() |* |* Beschreibung |* *************************************************************************/ SvHashTable::~SvHashTable() { #ifdef DOS_NIE printf( "Maximum: %ld, Fllung: %ld\n", (ULONG)nMax, (ULONG)nFill ); printf( "Anfragen: %ld, Versuche: %ld", (ULONG)lAsk, (ULONG)lTry ); if( lTry != 0 ) printf( ", V/E = %ld\n", lTry / lAsk ); #endif } /************************************************************************* |* |* SvHashTable::Test_Insert() |* |* Beschreibung |* *************************************************************************/ BOOL SvHashTable::Test_Insert( const void * pElement, BOOL bInsert, UINT32 * pInsertPos ) { UINT32 nHash; UINT32 nIndex; UINT32 nLoop; lAsk++; lTry++; nHash = HashFunc( pElement ); nIndex = nHash % nMax; // const char* s = ((ByteString*) pElement)->GetStr(); // fprintf(stderr,"### Hash: %lu , Name: %s\n",nIndex,s ); nLoop = 0; // divide to range while( (nMax != nLoop) && IsEntry( nIndex ) ) { // is place occupied if( COMPARE_EQUAL == Compare( pElement, nIndex ) ) { if( pInsertPos ) *pInsertPos = nIndex; // place of Element return TRUE; } nLoop++; lTry++; nIndex = (USHORT)(nIndex + nHash + 7) % nMax; } if( bInsert ) { DBG_ASSERT( nMax != nLoop, "Hash table full" ) if( nMax != nLoop ) { nFill++; *pInsertPos = nIndex; // return free place return TRUE; } } return( FALSE ); } /************************************************************************/ /************************************************************************* |* |* SvStringHashTable::SvStringHashTable() |* |* Beschreibung |* *************************************************************************/ SvStringHashTable::SvStringHashTable( UINT32 nMaxEntries ) : SvHashTable( nMaxEntries ) { #ifdef WIN DBG_ASSERT( (UINT32)nMaxEntries * sizeof( SvStringHashEntry ) <= 0xFF00, "Hash table size cannot exeed 64k byte" ) #endif pEntries = new SvStringHashEntry[ nMaxEntries ]; // RefCount auf eins setzen SvStringHashEntry * pPos, *pEnd; pPos = pEntries; pEnd = pEntries + nMaxEntries; while( pPos != pEnd ) { pPos->AddRef(); pPos++; } } /************************************************************************* |* |* ~SvStringHashTable::SvStringHashTable() |* |* Beschreibung |* *************************************************************************/ SvStringHashTable::~SvStringHashTable() { // RefCount auf eins setzen SvStringHashEntry * pPos, *pEnd; pPos = pEntries; pEnd = pEntries + GetMax(); #ifdef DBG_UTIL while( pPos != pEnd ) { DBG_ASSERT( pPos->GetRefCount() == 1, "Reference count != 1" ) pPos++; } #endif #ifdef MPW // der MPW-Compiler ruft sonst keine Dtoren! for ( USHORT n = 0; n < GetMax(); ++n ) (pEntries+n)->SvStringHashEntry::~SvStringHashEntry(); delete (void*) pEntries; #else delete [] pEntries; #endif } /************************************************************************* |* |* SvStringHashTable::HashFunc() |* |* Beschreibung |* *************************************************************************/ UINT32 SvStringHashTable::HashFunc( const void * pElement ) const { UINT32 nHash = 0; // hash value const char * pStr = ((const ByteString * )pElement)->GetBuffer(); int nShift = 0; while( *pStr ) { if( isupper( *pStr ) ) nHash ^= UINT32(*pStr - 'A' + 26) << nShift; else nHash ^= UINT32(*pStr - 'a') << nShift; if( nShift == 28 ) nShift = 0; else nShift += 4; pStr++; } return( nHash ); } /************************************************************************* |* |* SvStringHashTable::GetNearString() |* |* Beschreibung |* *************************************************************************/ ByteString SvStringHashTable::GetNearString( const ByteString & rName ) const { for( UINT32 i = 0; i < GetMax(); i++ ) { SvStringHashEntry * pE = Get( i ); if( pE ) { if( pE->GetName().EqualsIgnoreCaseAscii( rName ) && !pE->GetName().Equals( rName ) ) return pE->GetName(); } } return ByteString(); } /************************************************************************* |* |* SvStringHashTable::IsEntry() |* |* Beschreibung |* *************************************************************************/ BOOL SvStringHashTable::IsEntry( UINT32 nIndex ) const { if( nIndex >= GetMax() ) return FALSE; return pEntries[ nIndex ].HasId(); } /************************************************************************* |* |* SvStringHashTable::Insert() |* |* Beschreibung |* *************************************************************************/ BOOL SvStringHashTable::Insert( const ByteString & rName, UINT32 * pIndex ) { UINT32 nIndex; if( !pIndex ) pIndex = &nIndex; if( !SvHashTable::Test_Insert( &rName, TRUE, pIndex ) ) return FALSE; if( !IsEntry( *pIndex ) ) pEntries[ *pIndex ] = SvStringHashEntry( rName, *pIndex ); return TRUE; } /************************************************************************* |* |* SvStringHashTable::Test() |* |* Beschreibung |* *************************************************************************/ BOOL SvStringHashTable::Test( const ByteString & rName, UINT32 * pPos ) const { return ((SvStringHashTable *)this)->SvHashTable:: Test_Insert( &rName, FALSE, pPos ); } /************************************************************************* |* |* SvStringHashTable::Get() |* |* Beschreibung |* *************************************************************************/ SvStringHashEntry * SvStringHashTable::Get( UINT32 nIndex ) const { if( IsEntry( nIndex ) ) return pEntries + nIndex; return( NULL ); } /************************************************************************* |* |* SvStringHashTable::Get() |* |* Beschreibung |* *************************************************************************/ StringCompare SvStringHashTable::Compare( const void * pElement, UINT32 nIndex ) const { return ((const ByteString *)pElement)->CompareTo( pEntries[ nIndex ].GetName() ); } /************************************************************************* |* |* SvStringHashTable::FillHashList() |* |* Beschreibung |* *************************************************************************/ void SvStringHashTable::FillHashList( SvStringHashList * pList ) const { for( UINT32 n = 0; n < GetMax(); n++ ) { if( IsEntry( n ) ) pList->Insert( Get( n ), LIST_APPEND ); } // Hash Reihenfolge, jetzt sortieren }