Files
Yudong Jin 7a78369e4c Migrate to Zensical (#1869)
* Fix Russian Ruby code extraction.

* Add zensical configs.
2026-03-29 05:41:25 +08:00

148 lines
5.8 KiB
Ruby

=begin
File: hash_map_open_addressing.rb
Created Time: 2024-04-13
Author: Xuan Khoa Tu Nguyen (ngxktuzkai2000@gmail.com)
=end
require_relative './array_hash_map'
### Хеш-таблица с открытой адресацией ###
class HashMapOpenAddressing
TOMBSTONE = Pair.new(-1, '-1') # Удалить метку
### Конструктор ###
def initialize
@size = 0 # Число пар ключ-значение
@capacity = 4 # Вместимость хеш-таблицы
@load_thres = 2.0 / 3.0 # Порог коэффициента загрузки для запуска расширения
@extend_ratio = 2 # Коэффициент расширения
@buckets = Array.new(@capacity) # Массив корзин
end
### Хеш-функция ###
def hash_func(key)
key % @capacity
end
### Коэффициент загрузки ###
def load_factor
@size / @capacity
end
### Найти индекс корзины, соответствующий key ###
def find_bucket(key)
index = hash_func(key)
first_tombstone = -1
# Выполнять линейное пробирование и завершить при встрече с пустой корзиной
while !@buckets[index].nil?
# Если встретился key, вернуть соответствующий индекс корзины
if @buckets[index].key == key
# Если ранее встретилась метка удаления, переместить пару ключ-значение на этот индекс
if first_tombstone != -1
@buckets[first_tombstone] = @buckets[index]
@buckets[index] = TOMBSTONE
return first_tombstone # Вернуть индекс корзины после перемещения
end
return index # Вернуть индекс корзины
end
# Записать первую встретившуюся метку удаления
first_tombstone = index if first_tombstone == -1 && @buckets[index] == TOMBSTONE
# Вычислить индекс корзины; при выходе за конец вернуться к началу
index = (index + 1) % @capacity
end
# Если key не существует, вернуть индекс точки добавления
first_tombstone == -1 ? index : first_tombstone
end
### Операция поиска ###
def get(key)
# Найти индекс корзины, соответствующий key
index = find_bucket(key)
# Если пара ключ-значение найдена, вернуть соответствующее val
return @buckets[index].val unless [nil, TOMBSTONE].include?(@buckets[index])
# Если пара ключ-значение не существует, вернуть nil
nil
end
### Операция добавления ###
def put(key, val)
# Когда коэффициент загрузки превышает порог, выполнить расширение
extend if load_factor > @load_thres
# Найти индекс корзины, соответствующий key
index = find_bucket(key)
# Если пара ключ-значение найдена, перезаписать val и вернуть
unless [nil, TOMBSTONE].include?(@buckets[index])
@buckets[index].val = val
return
end
# Если пары ключ-значение нет, добавить ее
@buckets[index] = Pair.new(key, val)
@size += 1
end
### Операция удаления ###
def remove(key)
# Найти индекс корзины, соответствующий key
index = find_bucket(key)
# Если пара ключ-значение найдена, заменить ее меткой удаления
unless [nil, TOMBSTONE].include?(@buckets[index])
@buckets[index] = TOMBSTONE
@size -= 1
end
end
### Расширение хеш-таблицы ###
def extend
# Временно сохранить исходную хеш-таблицу
buckets_tmp = @buckets
# Инициализация новой хеш-таблицы после расширения
@capacity *= @extend_ratio
@buckets = Array.new(@capacity)
@size = 0
# Перенести пары ключ-значение из исходной хеш-таблицы в новую
for pair in buckets_tmp
put(pair.key, pair.val) unless [nil, TOMBSTONE].include?(pair)
end
end
### Вывести хеш-таблицу ###
def print
for pair in @buckets
if pair.nil?
puts "Nil"
elsif pair == TOMBSTONE
puts "TOMBSTONE"
else
puts "#{pair.key} -> #{pair.val}"
end
end
end
end
### Driver Code ###
if __FILE__ == $0
# Инициализация хеш-таблицы
hashmap = HashMapOpenAddressing.new
# Операция добавления
# Добавить пару (key, val) в хеш-таблицу
hashmap.put(12836, "Сяо Ха")
hashmap.put(15937, "Сяо Ло")
hashmap.put(16750, "Сяо Суань")
hashmap.put(13276, "Сяо Фа")
hashmap.put(10583, "Сяо Я")
puts "\nПосле добавления хеш-таблица имеет вид\nКлюч -> Значение"
hashmap.print
# Операция поиска
# Передать ключ key в хеш-таблицу и получить значение val
name = hashmap.get(13276)
puts "\nДля номера 13276 найдено имя #{name}"
# Операция удаления
# Удалить пару (key, val) из хеш-таблицы
hashmap.remove(16750)
puts "\nПосле удаления 16750 хеш-таблица имеет вид\nКлюч -> Значение"
hashmap.print
end