identifier.cpp

来自「konqueror3 embedded版本, KDE环境下的当家浏览器的嵌入式版」· C++ 代码 · 共 308 行

CPP
308
字号
/* *  This file is part of the KDE libraries *  Copyright (C) 2003 Apple Computer, Inc * *  This library is free software; you can redistribute it and/or *  modify it under the terms of the GNU Library General Public *  License as published by the Free Software Foundation; either *  version 2 of the License, or (at your option) any later version. * *  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 *  Library General Public License for more details. * *  You should have received a copy of the GNU Library General Public License *  along with this library; see the file COPYING.LIB.  If not, write to *  the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, *  Boston, MA 02110-1301, USA. * */#include "identifier.h"#include <stdio.h>#include <stdlib.h>#include <string.h>#define DUMP_STATISTICS 0namespace KJS {#if DUMP_STATISTICSstatic int numProbes;static int numCollisions;struct IdentifierStatisticsExitLogger { ~IdentifierStatisticsExitLogger(); };static IdentifierStatisticsExitLogger logger;IdentifierStatisticsExitLogger::~IdentifierStatisticsExitLogger(){    printf("\nKJS::Identifier statistics\n\n");    printf("%d probes\n", numProbes);    printf("%d collisions (%.1f%%)\n", numCollisions, 100.0 * numCollisions / numProbes);}#endifextern const Identifier argumentsPropertyName("arguments");extern const Identifier calleePropertyName("callee");extern const Identifier constructorPropertyName("constructor");extern const Identifier lengthPropertyName("length");extern const Identifier messagePropertyName("message");extern const Identifier namePropertyName("name");extern const Identifier prototypePropertyName("prototype");extern const Identifier specialPrototypePropertyName("__proto__");extern const Identifier toLocaleStringPropertyName("toLocaleString");extern const Identifier toStringPropertyName("toString");extern const Identifier valueOfPropertyName("valueOf");static const int _minTableSize = 64;UString::Rep **Identifier::_table;int Identifier::_tableSize;int Identifier::_tableSizeMask;int Identifier::_keyCount;bool Identifier::equal(UString::Rep *r, const char *s){    int length = r->len;    const UChar *d = r->data();    for (int i = 0; i != length; ++i)        if (d[i].uc != (unsigned char)s[i])            return false;    return s[length] == 0;}bool Identifier::equal(UString::Rep *r, const UChar *s, int length){    if (r->len != length)        return false;    const UChar *d = r->data();    for (int i = 0; i != length; ++i)        if (d[i].uc != s[i].uc)            return false;    return true;}bool Identifier::equal(UString::Rep *r, UString::Rep *b){    int length = r->len;    if (length != b->len)        return false;    const UChar *d = r->data();    const UChar *s = b->data();    for (int i = 0; i != length; ++i)        if (d[i].uc != s[i].uc)            return false;    return true;}UString::Rep *Identifier::add(const char *c){    if (!c)        return &UString::Rep::null;    int length = strlen(c);    if (length == 0)        return &UString::Rep::empty;    if (!_table)        expand();    unsigned hash = UString::Rep::computeHash(c);    int i = hash & _tableSizeMask;#if DUMP_STATISTICS    ++numProbes;    numCollisions += _table[i] && !equal(_table[i], c);#endif    while (UString::Rep *key = _table[i]) {        if (equal(key, c))            return key;        i = (i + 1) & _tableSizeMask;    }    UChar *d = new UChar[length];    for (int j = 0; j != length; j++)        d[j] = c[j];    UString::Rep *r = new UString::Rep;    r->dat = d;    r->len = length;    r->capacity = UString::Rep::capacityForIdentifier;    r->rc = 0;    r->_hash = hash;    _table[i] = r;    ++_keyCount;    if (_keyCount * 2 >= _tableSize)        expand();    return r;}UString::Rep *Identifier::add(const UChar *s, int length){    if (length == 0)        return &UString::Rep::empty;    if (!_table)        expand();    unsigned hash = UString::Rep::computeHash(s, length);    int i = hash & _tableSizeMask;#if DUMP_STATISTICS    ++numProbes;    numCollisions += _table[i] && !equal(_table[i], s, length);#endif    while (UString::Rep *key = _table[i]) {        if (equal(key, s, length))            return key;        i = (i + 1) & _tableSizeMask;    }    UChar *d = new UChar[length];    for (int j = 0; j != length; j++)        d[j] = s[j];    UString::Rep *r = new UString::Rep;    r->dat = d;    r->len = length;    r->capacity = UString::Rep::capacityForIdentifier;    r->rc = 0;    r->_hash = hash;    _table[i] = r;    ++_keyCount;    if (_keyCount * 2 >= _tableSize)        expand();    return r;}UString::Rep *Identifier::add(UString::Rep *r){    if (r->capacity == UString::Rep::capacityForIdentifier)        return r;    if (r->len == 0)        return &UString::Rep::empty;    if (!_table)        expand();    unsigned hash = r->hash();    int i = hash & _tableSizeMask;#if DUMP_STATISTICS    ++numProbes;    numCollisions += _table[i] && !equal(_table[i], r);#endif    while (UString::Rep *key = _table[i]) {        if (equal(key, r))            return key;        i = (i + 1) & _tableSizeMask;    }    r->capacity = UString::Rep::capacityForIdentifier;    _table[i] = r;    ++_keyCount;    if (_keyCount * 2 >= _tableSize)        expand();    return r;}inline void Identifier::insert(UString::Rep *key){    unsigned hash = key->hash();    int i = hash & _tableSizeMask;#if DUMP_STATISTICS    ++numProbes;    numCollisions += _table[i] != 0;#endif    while (_table[i])        i = (i + 1) & _tableSizeMask;    _table[i] = key;}void Identifier::remove(UString::Rep *r){    unsigned hash = r->hash();    UString::Rep *key;    int i = hash & _tableSizeMask;#if DUMP_STATISTICS    ++numProbes;    numCollisions += _table[i] && equal(_table[i], r);#endif    while ((key = _table[i])) {        if (equal(key, r))            break;        i = (i + 1) & _tableSizeMask;    }    if (!key)        return;    _table[i] = 0;    --_keyCount;    if (_keyCount * 6 < _tableSize && _tableSize > _minTableSize) {        shrink();        return;    }    // Reinsert all the items to the right in the same cluster.    while (1) {        i = (i + 1) & _tableSizeMask;        key = _table[i];        if (!key)            break;        _table[i] = 0;        insert(key);    }}void Identifier::expand(){    rehash(_tableSize == 0 ? _minTableSize : _tableSize * 2);}void Identifier::shrink(){    rehash(_tableSize / 2);}void Identifier::rehash(int newTableSize){    int oldTableSize = _tableSize;    UString::Rep **oldTable = _table;    _tableSize = newTableSize;    _tableSizeMask = newTableSize - 1;    _table = (UString::Rep **)calloc(newTableSize, sizeof(UString::Rep *));    for (int i = 0; i != oldTableSize; ++i)        if (UString::Rep *key = oldTable[i])            insert(key);    free(oldTable);}const Identifier &Identifier::null(){    static Identifier null;    return null;}} // namespace KJS

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?