Как удалить ключ из хэша и получить оставшийся хэш в 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} 

почему бы просто не использовать:

hash.delete(key)

есть много способов удалить ключ из хэша и получить оставшийся хэш в Ruby.

  1. .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} 
    
  2. .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} 
    
  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}         
    
  4. .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

в чистом Руби:

{:a => 1, :b => 2}.tap{|x| x.delete(:a)}   # => {:b=>2}

посмотреть 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} 

Это также будет работать: hash[hey] = nil