227 lines
7.6 KiB
C#
227 lines
7.6 KiB
C#
|
//------------------------------------------------------------------------------
|
||
|
// <copyright file="NameTable.cs" company="Microsoft">
|
||
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
// </copyright>
|
||
|
// <owner current="true" primary="true">[....]</owner>
|
||
|
//------------------------------------------------------------------------------
|
||
|
|
||
|
using System;
|
||
|
|
||
|
namespace System.Xml {
|
||
|
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// <para>
|
||
|
/// XmlNameTable implemented as a simple hash table.
|
||
|
/// </para>
|
||
|
/// </devdoc>
|
||
|
public class NameTable : XmlNameTable {
|
||
|
//
|
||
|
// Private types
|
||
|
//
|
||
|
class Entry {
|
||
|
internal string str;
|
||
|
internal int hashCode;
|
||
|
internal Entry next;
|
||
|
|
||
|
internal Entry( string str, int hashCode, Entry next ) {
|
||
|
this.str = str;
|
||
|
this.hashCode = hashCode;
|
||
|
this.next = next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Fields
|
||
|
//
|
||
|
Entry[] entries;
|
||
|
int count;
|
||
|
int mask;
|
||
|
int hashCodeRandomizer;
|
||
|
|
||
|
//
|
||
|
// Constructor
|
||
|
//
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable.NameTable"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// Public constructor.
|
||
|
/// </devdoc>
|
||
|
public NameTable() {
|
||
|
mask = 31;
|
||
|
entries = new Entry[mask+1];
|
||
|
hashCodeRandomizer = Environment.TickCount;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// XmlNameTable public methods
|
||
|
//
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable.Add"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// Add the given string to the NameTable or return
|
||
|
/// the existing string if it is already in the NameTable.
|
||
|
/// </devdoc>
|
||
|
public override string Add( string key ) {
|
||
|
if ( key == null ) {
|
||
|
throw new ArgumentNullException( "key" );
|
||
|
}
|
||
|
int len = key.Length;
|
||
|
if ( len == 0 ) {
|
||
|
return string.Empty;
|
||
|
}
|
||
|
int hashCode = len + hashCodeRandomizer;
|
||
|
// use key.Length to eliminate the rangecheck
|
||
|
for ( int i = 0; i < key.Length; i++ ) {
|
||
|
hashCode += ( hashCode << 7 ) ^ key[i];
|
||
|
}
|
||
|
// mix it a bit more
|
||
|
hashCode -= hashCode >> 17;
|
||
|
hashCode -= hashCode >> 11;
|
||
|
hashCode -= hashCode >> 5;
|
||
|
|
||
|
for ( Entry e = entries[hashCode & mask]; e != null; e = e.next ) {
|
||
|
if ( e.hashCode == hashCode && e.str.Equals( key ) ) {
|
||
|
return e.str;
|
||
|
}
|
||
|
}
|
||
|
return AddEntry( key, hashCode );
|
||
|
}
|
||
|
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable.Add1"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// Add the given string to the NameTable or return
|
||
|
/// the existing string if it is already in the NameTable.
|
||
|
/// </devdoc>
|
||
|
public override string Add( char[] key, int start, int len ) {
|
||
|
if ( len == 0 ) {
|
||
|
return string.Empty;
|
||
|
}
|
||
|
|
||
|
int hashCode = len + hashCodeRandomizer;
|
||
|
hashCode += ( hashCode << 7 ) ^ key[start]; // this will throw IndexOutOfRangeException in case the start index is invalid
|
||
|
int end = start+len;
|
||
|
for ( int i = start + 1; i < end; i++) {
|
||
|
hashCode += ( hashCode << 7 ) ^ key[i];
|
||
|
}
|
||
|
// mix it a bit more
|
||
|
hashCode -= hashCode >> 17;
|
||
|
hashCode -= hashCode >> 11;
|
||
|
hashCode -= hashCode >> 5;
|
||
|
|
||
|
for ( Entry e = entries[hashCode & mask]; e != null; e = e.next ) {
|
||
|
if ( e.hashCode == hashCode && TextEquals( e.str, key, start, len ) ) {
|
||
|
return e.str;
|
||
|
}
|
||
|
}
|
||
|
return AddEntry( new string( key, start, len ), hashCode );
|
||
|
}
|
||
|
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable.Get"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// Find the matching string in the NameTable.
|
||
|
/// </devdoc>
|
||
|
public override string Get( string value ) {
|
||
|
if ( value == null ) {
|
||
|
throw new ArgumentNullException("value");
|
||
|
}
|
||
|
if ( value.Length == 0 ) {
|
||
|
return string.Empty;
|
||
|
}
|
||
|
|
||
|
int len = value.Length + hashCodeRandomizer;
|
||
|
int hashCode = len;
|
||
|
// use value.Length to eliminate the rangecheck
|
||
|
for ( int i = 0; i < value.Length; i++ ) {
|
||
|
hashCode += ( hashCode << 7 ) ^ value[i];
|
||
|
}
|
||
|
// mix it a bit more
|
||
|
hashCode -= hashCode >> 17;
|
||
|
hashCode -= hashCode >> 11;
|
||
|
hashCode -= hashCode >> 5;
|
||
|
|
||
|
for ( Entry e = entries[hashCode & mask]; e != null; e = e.next ) {
|
||
|
if ( e.hashCode == hashCode && e.str.Equals( value ) ) {
|
||
|
return e.str;
|
||
|
}
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
/// <include file='doc\NameTable.uex' path='docs/doc[@for="NameTable.Get1"]/*' />
|
||
|
/// <devdoc>
|
||
|
/// Find the matching string atom given a range of
|
||
|
/// characters.
|
||
|
/// </devdoc>
|
||
|
public override string Get( char[] key, int start, int len ) {
|
||
|
if ( len == 0 ) {
|
||
|
return string.Empty;
|
||
|
}
|
||
|
|
||
|
int hashCode = len + hashCodeRandomizer;
|
||
|
hashCode += ( hashCode << 7 ) ^ key[start]; // this will throw IndexOutOfRangeException in case the start index is invalid
|
||
|
int end = start+len;
|
||
|
for ( int i = start + 1; i < end; i++) {
|
||
|
hashCode += ( hashCode << 7 ) ^ key[i];
|
||
|
}
|
||
|
// mix it a bit more
|
||
|
hashCode -= hashCode >> 17;
|
||
|
hashCode -= hashCode >> 11;
|
||
|
hashCode -= hashCode >> 5;
|
||
|
|
||
|
for ( Entry e = entries[hashCode & mask]; e != null; e = e.next ) {
|
||
|
if ( e.hashCode == hashCode && TextEquals( e.str, key, start, len ) ) {
|
||
|
return e.str;
|
||
|
}
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Private methods
|
||
|
//
|
||
|
|
||
|
private string AddEntry( string str, int hashCode ) {
|
||
|
int index = hashCode & mask;
|
||
|
Entry e = new Entry( str, hashCode, entries[index] );
|
||
|
entries[index] = e;
|
||
|
if ( count++ == mask ) {
|
||
|
Grow();
|
||
|
}
|
||
|
return e.str;
|
||
|
}
|
||
|
|
||
|
private void Grow() {
|
||
|
int newMask = mask * 2 + 1;
|
||
|
Entry[] oldEntries = entries;
|
||
|
Entry[] newEntries = new Entry[newMask+1];
|
||
|
|
||
|
// use oldEntries.Length to eliminate the rangecheck
|
||
|
for ( int i = 0; i < oldEntries.Length; i++ ) {
|
||
|
Entry e = oldEntries[i];
|
||
|
while ( e != null ) {
|
||
|
int newIndex = e.hashCode & newMask;
|
||
|
Entry tmp = e.next;
|
||
|
e.next = newEntries[newIndex];
|
||
|
newEntries[newIndex] = e;
|
||
|
e = tmp;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
entries = newEntries;
|
||
|
mask = newMask;
|
||
|
}
|
||
|
|
||
|
private static bool TextEquals( string str1, char[] str2, int str2Start, int str2Length ) {
|
||
|
if ( str1.Length != str2Length ) {
|
||
|
return false;
|
||
|
}
|
||
|
// use array.Length to eliminate the rangecheck
|
||
|
for ( int i = 0; i < str1.Length; i++ ) {
|
||
|
if ( str1[i] != str2[str2Start+i] ) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
}
|