Как сравнить два хэша?
Я пытаюсь сравнить два рубиновых хэша, используя следующий код:
#!/usr/bin/env ruby
require "yaml"
require "active_support"
file1 = YAML::load(File.open('./en_20110207.yml'))
file2 = YAML::load(File.open('./locales/en.yml'))
arr = []
file1.select { |k,v|
file2.select { |k2, v2|
arr << "#{v2}" if "#{v}" != "#{v2}"
}
}
puts arr
вывод на экран-это полный файл из file2. Я точно знаю, что файлы разные, но сценарий, похоже, не поднимает его.
11 ответов:
вы можете сравнить хэши прямо для равенства:
hash1 = {'a' => 1, 'b' => 2} hash2 = {'a' => 1, 'b' => 2} hash3 = {'a' => 1, 'b' => 2, 'c' => 3} hash1 == hash2 # => true hash1 == hash3 # => false hash1.to_a == hash2.to_a # => true hash1.to_a == hash3.to_a # => false
вы можете конвертировать хэши в массивы, а затем получить их разницу:
hash3.to_a - hash1.to_a # => [["c", 3]] if (hash3.size > hash1.size) difference = hash3.to_a - hash1.to_a else difference = hash1.to_a - hash3.to_a end Hash[*difference.flatten] # => {"c"=>3}
упрощения в дальнейшем:
назначение разности через тернарную структуру:
difference = (hash3.size > hash1.size) \ ? hash3.to_a - hash1.to_a \ : hash1.to_a - hash3.to_a => [["c", 3]] Hash[*difference.flatten] => {"c"=>3}делать все это в одной операции и избавление от
differenceпеременной:Hash[*( (hash3.size > hash1.size) \ ? hash3.to_a - hash1.to_a \ : hash1.to_a - hash3.to_a ).flatten] => {"c"=>3}
вы можете попробовать hashdiff gem, который позволяет глубокое сравнение хэшей и массивов в хэше.
пример:
a = {a:{x:2, y:3, z:4}, b:{x:3, z:45}} b = {a:{y:3}, b:{y:3, z:30}} diff = HashDiff.diff(a, b) diff.should == [['-', 'a.x', 2], ['-', 'a.z', 4], ['-', 'b.x', 3], ['~', 'b.z', 45, 30], ['+', 'b.y', 3]]
Если вы хотите получить, в чем разница между двумя хэшами, вы можете сделать это:
h1 = {:a => 20, :b => 10, :c => 44} h2 = {:a => 2, :b => 10, :c => "44"} result = {} h1.each {|k, v| result[k] = h2[k] if h2[k] != v } p result #=> {:a => 2, :c => "44"}
рельсы осужденный the
diffметод.для быстрого одну строчку:
hash1.to_s == hash2.to_s
вы можете использовать простое пересечение массива, таким образом, вы можете знать, что отличается в каждом хэше.
hash1 = { a: 1 , b: 2 } hash2 = { a: 2 , b: 2 } overlapping_elements = hash1.to_a & hash2.to_a exclusive_elements_from_hash1 = hash1.to_a - overlapping_elements exclusive_elements_from_hash2 = hash2.to_a - overlapping_elements
на это ответили в " сравнение рубиновых хэшей". Рельсы добавляет
diffметод в хэши. Это хорошо работает.
У меня была та же проблема и отправили запрос на рельсы
- работает с глубоко вложенных хэш -
- работает с массивами хэшей
https://github.com/elfassy/rails/commit/5f3410e04fe8c4d4745397db866c9633b80ccec6
Если вам нужна быстрая и грязная разница между хэшами, которая правильно поддерживает nil в значениях, вы можете использовать что-то вроде
def diff(one, other) (one.keys + other.keys).uniq.inject({}) do |memo, key| unless one.key?(key) && other.key?(key) && one[key] == other[key] memo[key] = [one.key?(key) ? one[key] : :_no_key, other.key?(key) ? other[key] : :_no_key] end memo end end
Если вы хотите красиво отформатированный diff, вы можете сделать это:
# Gemfile gem 'awesome_print' # or gem install awesome_printи в коде:
require 'ap' def my_diff(a, b) as = a.ai(plain: true).split("\n").map(&:strip) bs = b.ai(plain: true).split("\n").map(&:strip) ((as - bs) + (bs - as)).join("\n") end puts my_diff({foo: :bar, nested: {val1: 1, val2: 2}, end: :v}, {foo: :bar, n2: {nested: {val1: 1, val2: 3}}, end: :v})идея состоит в том, чтобы использовать удивительную печать для форматирования и дифференцировать вывод. Разница не будет точной, но она полезна для целей отладки.
... а теперь в модуль форма для применения к различным классам коллекций (хэш среди них). Это не глубокий осмотр, но это просто.
# Enable "diffing" and two-way transformations between collection objects module Diffable # Calculates the changes required to transform self to the given collection. # @param b [Enumerable] The other collection object # @return [Array] The Diff: A two-element change set representing items to exclude and items to include def diff( b ) a, b = to_a, b.to_a [a - b, b - a] end # Consume return value of Diffable#diff to produce a collection equal to the one used to produce the given diff. # @param to_drop [Enumerable] items to exclude from the target collection # @param to_add [Enumerable] items to include in the target collection # @return [Array] New transformed collection equal to the one used to create the given change set def apply_diff( to_drop, to_add ) to_a - to_drop + to_add end end if __FILE__ == # Demo: Hashes with overlapping keys and somewhat random values. Hash.send :include, Diffable rng = Random.new a = (:a..:q).to_a.reduce(Hash[]){|h,k| h.merge! Hash[k, rng.rand(2)] } b = (:i..:z).to_a.reduce(Hash[]){|h,k| h.merge! Hash[k, rng.rand(2)] } raise unless a == Hash[ b.apply_diff(*b.diff(a)) ] # change b to a raise unless b == Hash[ a.apply_diff(*a.diff(b)) ] # change a to b raise unless a == Hash[ a.apply_diff(*a.diff(a)) ] # change a to a raise unless b == Hash[ b.apply_diff(*b.diff(b)) ] # change b to b end
Comments