Avoid rehashing in Hash#select/reject [Bug #16996]

This commit is contained in:
Marc-Andre Lafortune 2021-03-15 23:51:13 -04:00
Родитель 85f99f4b71
Коммит d094c3ef04
2 изменённых файлов: 32 добавлений и 25 удалений

32
hash.c
Просмотреть файл

@ -2547,15 +2547,6 @@ rb_hash_reject_bang(VALUE hash)
return hash;
}
static int
reject_i(VALUE key, VALUE value, VALUE result)
{
if (!RTEST(rb_yield_values(2, key, value))) {
rb_hash_aset(result, key, value);
}
return ST_CONTINUE;
}
/*
* call-seq:
* hash.reject {|key, value| ... } -> new_hash
@ -2586,9 +2577,9 @@ rb_hash_reject(VALUE hash)
rb_warn("extra states are no longer copied: %+"PRIsVALUE, hash);
}
}
result = rb_hash_new();
result = hash_copy(hash_alloc(rb_cHash), hash);
if (!RHASH_EMPTY_P(hash)) {
rb_hash_foreach(hash, reject_i, result);
rb_hash_foreach(result, delete_if_i, result);
}
return result;
}
@ -2711,10 +2702,10 @@ rb_hash_fetch_values(int argc, VALUE *argv, VALUE hash)
}
static int
select_i(VALUE key, VALUE value, VALUE result)
keep_if_i(VALUE key, VALUE value, VALUE hash)
{
if (RTEST(rb_yield_values(2, key, value))) {
rb_hash_aset(result, key, value);
if (!RTEST(rb_yield_values(2, key, value))) {
return ST_DELETE;
}
return ST_CONTINUE;
}
@ -2742,22 +2733,13 @@ rb_hash_select(VALUE hash)
VALUE result;
RETURN_SIZED_ENUMERATOR(hash, 0, 0, hash_enum_size);
result = rb_hash_new();
result = hash_copy(hash_alloc(rb_cHash), hash);
if (!RHASH_EMPTY_P(hash)) {
rb_hash_foreach(hash, select_i, result);
rb_hash_foreach(result, keep_if_i, result);
}
return result;
}
static int
keep_if_i(VALUE key, VALUE value, VALUE hash)
{
if (!RTEST(rb_yield_values(2, key, value))) {
return ST_DELETE;
}
return ST_CONTINUE;
}
/*
* call-seq:
* hash.select! {|key, value| ... } -> self or nil

Просмотреть файл

@ -122,6 +122,31 @@ class TestHash < Test::Unit::TestCase
assert_equal set2, set2.dup
end
def assert_hash_does_not_rehash
obj = Object.new
class << obj
attr_accessor :hash_calls
def hash
@hash_calls += 1
super
end
end
obj.hash_calls = 0
hash = {obj => 42}
assert_equal(1, obj.hash_calls)
yield hash
assert_equal(1, obj.hash_calls)
end
def test_select_reject_will_not_rehash
assert_hash_does_not_rehash do |hash|
hash.select { true }
end
assert_hash_does_not_rehash do |hash|
hash.reject { false }
end
end
def test_s_AREF
h = @cls["a" => 100, "b" => 200]
assert_equal(100, h['a'])