zoukankan      html  css  js  c++  java
  • 字典(python)

    # -*- coding: utf-8 -*-
    
    class Array(object):
    
        def __init__(self, size=32, init=None):
            self._size = size
            self._items = [init] * size
    
        def __getitem__(self, index):
            return self._items[index]
    
        def __setitem__(self, index, value):
            self._items[index] = value
    
        def __len__(self):
            return self._size
    
        def clear(self, value=None):
            for i in range(len(self._items)):
                self._items[i] = value
    
        def __iter__(self):
            for item in self._items:
                yield item
    
    
    class Slot(object):
    
        def __init__(self, key, value):
            self.key, self.value = key, value
    
    
    class HashTable(object):
    
        UNUSED = None  # 没被使用过
        EMPTY = Slot(None, None)  # 使用却被删除过
    
        def __init__(self):
            self._table = Array(8, init=HashTable.UNUSED)   # 保持 2*i 次方
            self.length = 0
    
        @property
        def _load_factor(self):
            # load_factor 超过 0.8 重新分配
            return self.length / float(len(self._table))
    
        def __len__(self):
            return self.length
    
        def _hash(self, key):
            return abs(hash(key)) % len(self._table)
    
        def _find_key(self, key):
            index = self._hash(key)
            _len = len(self._table)
            while self._table[index] is not HashTable.UNUSED:
                if self._table[index] is HashTable.EMPTY:
                    index = (index*5 + 1) % _len
                    continue
                elif self._table[index].key == key:
                    return index
                else:
                    index = (index*5 + 1) % _len
            return None
    
        def _find_slot_for_insert(self, key):
            index = self._hash(key)
            _len = len(self._table)
            while not self._slot_can_insert(index):
                index = (index*5 + 1) % _len
            return index
    
        def _slot_can_insert(self, index):
            return (self._table[index] is HashTable.EMPTY or self._table[index] is HashTable.UNUSED)
    
        def __contains__(self, key):  # in operator
            index = self._find_key(key)
            return index is not None
    
        def add(self, key, value):
            if key in self:
                index = self._find_key(key)
                self._table[index].value = value
                return False
            else:
                index = self._find_slot_for_insert(key)
                self._table[index] = Slot(key, value)
                self.length += 1
                if self._load_factor >= 0.8:
                    self._rehash()
                return True
    
        def _rehash(self):
            old_table = self._table
            newsize = len(self._table) * 2
            self._table = Array(newsize, HashTable.UNUSED)
    
            self.length = 0
    
            for slot in old_table:
                if slot is not HashTable.UNUSED and slot is not HashTable.EMPTY:
                    index = self._find_slot_for_insert(slot.key)
                    self._table[index] = slot
                    self.length += 1
    
        def get(self, key, default=None):
            index = self._find_key(key)
            if index is None:
                return default
            else:
                return self._table[index].value
    
        def remove(self, key):
            index = self._find_key(key)
            if index is None:
                raise KeyError()
            value = self._table[index].value
            self.length -= 1
            self._table[index] = HashTable.EMPTY
            return value
    
        def __iter__(self):
            for slot in self._table:
                if slot not in (HashTable.EMPTY, HashTable.UNUSED):
                    yield slot.key
    
    class DictADT(HashTable):
    
        def _iter_slot(self):
            for slot in self._table:
                if slot not in (HashTable.EMPTY, HashTable.UNUSED):
                    yield slot
    
        def __setitem__(self, key, value):
            self.add(key, value)
    
        def __getitem__(self, key):
            if key not in self:
                raise KeyError()
            else:
                return self.get(key)
    
        def items(self):
            for slot in self._iter_slot():
                yield (slot.key, slot.value)
    
        def keys(self):
            for slot in self._iter_slot():
                yield slot.key
    
        def values(self):
            for slot in self._iter_slot():
                yield slot.value
    
    
    def test_dict_adt():
        import random
        d = DictADT()
    
        d['a'] = 1
        assert d['a'] == 1
        d.remove('a')
    
        l = list(range(30))
        random.shuffle(l)
        for i in l:
            d.add(i, i)
    
        for i in range(30):
            assert d.get(i) == i
    
        assert sorted(list(d.keys())) == sorted(l)
    
    
    test_dict_adt()
  • 相关阅读:
    浅谈值对象
    循环一个节点列表(NodeList)或者数组,并且绑定事件处理函数引发对闭包的理解
    当前窗口和Iframe之间的相互访问(图片上传成功后立刻显示在当前页面上)
    网页动态加载图片 通过JS和jquery实现。
    javascript拖动层函数封装
    javascript中变量声明提升(Hoisting)
    运动框架必备的运动算法 留着用!
    CSS3特性之改变在浏览器上选中文字时,默认的背景颜色和文字颜色
    仿淘宝商品图片放大镜效果(鼠标移动上去会出现放大的图片,并且可以移动)
    与PHP交互中文编码的几个函数 decodeURIComponent,encodeURIComponent,encodeURI,decodeURI
  • 原文地址:https://www.cnblogs.com/muzinan110/p/11166946.html
Copyright © 2011-2022 走看看