Как удалить ключ из хэша и получить оставшийся хэш в Ruby/Rails?
чтобы добавить новую пару в хэш, я делаю:
{:a => 1, :b => 2}.merge!({:c => 3}) #=> {:a => 1, :b => 2, :c => 3}
есть ли аналогичный способ удалить ключ из хэша ?
это работает:
{:a => 1, :b => 2}.reject! { |k| k == :a } #=> {:b => 2}
но я ожидал бы иметь что-то вроде:
{:a => 1, :b => 2}.delete!(:a) #=> {:b => 2}
важно, чтобы возвращаемое значение было оставшимся хэшем, поэтому я мог бы делать такие вещи, как:
foo(my_hash.reject! { |k| k == my_key })
в одну строку.
13 ответов
Rails имеет исключение / исключение! метод который возвращает хэш с удаленными ключами. Если вы уже используете Rails, нет смысла создавать свою собственную версию этого.
class Hash
# Returns a hash that includes everything but the given keys.
# hash = { a: true, b: false, c: nil}
# hash.except(:c) # => { a: true, b: false}
# hash # => { a: true, b: false, c: nil}
#
# This is useful for limiting a set of parameters to everything but a few known toggles:
# @person.update(params[:person].except(:admin))
def except(*keys)
dup.except!(*keys)
end
# Replaces the hash without the given keys.
# hash = { a: true, b: false, c: nil}
# hash.except!(:c) # => { a: true, b: false}
# hash # => { a: true, b: false }
def except!(*keys)
keys.each { |key| delete(key) }
self
end
end
oneliner plain ruby, он работает только с ruby > 1.9.x:
1.9.3p0 :002 > h = {:a => 1, :b => 2}
=> {:a=>1, :b=>2}
1.9.3p0 :003 > h.tap { |hs| hs.delete(:a) }
=> {:b=>2}
нажми метод всегда возвращает объект, для которого вызывается...
в противном случае, если вы требуете active_support/core_ext/hash
(что автоматически требовало в каждом приложении Rails) вы можете использовать один из следующих способов в зависимости от ваших потребностей:
➜ ~ irb
1.9.3p125 :001 > require 'active_support/core_ext/hash' => true
1.9.3p125 :002 > h = {:a => 1, :b => 2, :c => 3}
=> {:a=>1, :b=>2, :c=>3}
1.9.3p125 :003 > h.except(:a)
=> {:b=>2, :c=>3}
1.9.3p125 :004 > h.slice(:a)
=> {:a=>1}
за исключением использует подход черного списка, поэтому он удаляет все ключи, перечисленные как args, в то время как фрагмент использует подход белого списка, поэтому он удаляет все ключи, которые не перечислены в качестве аргументов. Существует также версия bang этого метода (except!
и slice!
), которые изменяют данный хэш, но их возвращаемое значение отличается, оба они возвращают хэш. Он представляет собой удаленные ключи для slice!
и ключи, которые хранятся на except!
:
1.9.3p125 :011 > {:a => 1, :b => 2, :c => 3}.except!(:a)
=> {:b=>2, :c=>3}
1.9.3p125 :012 > {:a => 1, :b => 2, :c => 3}.slice!(:a)
=> {:b=>2, :c=>3}
есть много способов удалить ключ из хэша и получить оставшийся хэш в Ruby.
-
.slice
=> он вернет выбранные ключи и не удалит их из исходного хэша2.2.2 :074 > hash = {"one"=>1, "two"=>2, "three"=>3} => {"one"=>1, "two"=>2, "three"=>3} 2.2.2 :075 > hash.slice("one","two") => {"one"=>1, "two"=>2} 2.2.2 :076 > hash => {"one"=>1, "two"=>2, "three"=>3}
-
.delete
=> он удалит выбранные ключи из исходного хэша (он может принимать только один ключ и не более одного)2.2.2 :094 > hash = {"one"=>1, "two"=>2, "three"=>3} => {"one"=>1, "two"=>2, "three"=>3} 2.2.2 :095 > hash.delete("one") => 1 2.2.2 :096 > hash => {"two"=>2, "three"=>3}
-
.except
=> он вернет оставшиеся ключи, но ничего не удалит из исходного хэша2.2.2 :097 > hash = {"one"=>1, "two"=>2, "three"=>3} => {"one"=>1, "two"=>2, "three"=>3} 2.2.2 :098 > hash.except("one","two") => {"three"=>3} 2.2.2 :099 > hash => {"one"=>1, "two"=>2, "three"=>3}
-
.delete_if
=> в случае, если вам необходимо удалить ключ на основе значения. Очевидно, что он удалит соответствующие ключи из исходного хэша2.2.2 :115 > hash = {"one"=>1, "two"=>2, "three"=>3, "one_again"=>1} => {"one"=>1, "two"=>2, "three"=>3, "one_again"=>1} 2.2.2 :116 > value = 1 => 1 2.2.2 :117 > hash.delete_if { |k,v| v == value } => {"two"=>2, "three"=>3} 2.2.2 :118 > hash => {"two"=>2, "three"=>3}
результаты на основе Ruby 2.2.2.
Если вы хотите использовать pure Ruby (без Rails), не хотите создавать методы расширения (возможно, Вам это нужно только в одном или двух местах и не хотите загрязнять пространство имен тоннами методов) и не хотите редактировать хэш на месте (т. е. вы поклонник функционального программирования, как я), вы можете "выбрать":
>> x = {:a => 1, :b => 2, :c => 3}
=> {:a=>1, :b=>2, :c=>3}
>> x.select{|x| x != :a}
=> {:b=>2, :c=>3}
>> x.select{|x| ![:a, :b].include?(x)}
=> {:c=>3}
>> x
=> {:a=>1, :b=>2, :c=>3}
#in lib/core_extensions.rb
class Hash
#pass single or array of keys, which will be removed, returning the remaining hash
def remove!(*keys)
keys.each{|key| self.delete(key) }
self
end
#non-destructive version
def remove(*keys)
self.dup.remove!(*keys)
end
end
#in config/initializers/app_environment.rb (or anywhere in config/initializers)
require 'core_extensions'
Я настроил это так ,что.remove возвращает копию хэша с удаленными ключами, а remove! изменяет сам хэш. Это соответствует условностям ruby. например, из консоли
>> hash = {:a => 1, :b => 2}
=> {:b=>2, :a=>1}
>> hash.remove(:a)
=> {:b=>2}
>> hash
=> {:b=>2, :a=>1}
>> hash.remove!(:a)
=> {:b=>2}
>> hash
=> {:b=>2}
>> hash.remove!(:a, :b)
=> {}
можно использовать except!
С facets
gem:
>> require 'facets' # or require 'facets/hash/except'
=> true
>> {:a => 1, :b => 2}.except(:a)
=> {:b=>2}
исходный хэш не меняется.
EDIT: как говорит Рассел, фасеты имеют некоторые скрытые проблемы и не полностью API-совместимы с ActiveSupport. С другой стороны, ActiveSupport не так полон, как фасеты. В конце концов, я бы использовал AS и позволил крайним случаям в вашем коде.
вместо исправления обезьян или ненужного включения больших библиотек вы можете использовать уточнения, если вы используете Ruby 2:
module HashExtensions
refine Hash do
def except!(*candidates)
candidates.each { |candidate| delete(candidate) }
self
end
def except(*candidates)
dup.remove!(candidates)
end
end
end
Вы можете использовать эту функцию, не влияя на другие части вашей программы, или включать большие внешние библиотеки.
class FabulousCode
using HashExtensions
def incredible_stuff
delightful_hash.except(:not_fabulous_key)
end
end
посмотреть Ruby on Rails: удалить несколько хэш-ключей
hash.delete_if{ |k,| keys_to_delete.include? k }
Это однострочный способ сделать это, но он не очень читаемый. Рекомендуется использовать две строки.
use_remaining_hash_for_something(Proc.new { hash.delete(:key); hash }.call)
было здорово, если delete вернет пару delete хэша. Я делаю это:
hash = {a: 1, b: 2, c: 3}
{b: hash.delete(:b)} # => {:b=>2}
hash # => {:a=>1, :c=>3}