2017-09-17 1 views
2

J'ai deux tableaux de hash comme ceci:Ruby: itérer et nid hachages à la condition

h1=[{id:1, item:1, from: DateTime.new(2017,9,4,6,0,0,'+0300'), to: DateTime.new(2017,9,4,17,59,59,'+0300'), value:10}, 
    {id:1, item:2, from: DateTime.new(2017,9,4,18,0,0,'+0300'), to: DateTime.new(2017,9,4,23,59,59,'+0300'), value:10}] 

h2=[{id:1, item:1, date: DateTime.new(2017,9,4,6,10,0,'+0300'), value:5}, 
    {id:2, item:1, date: DateTime.new(2017,9,4,7,20,0,'+0300'), value:7}, 
    {id:3, item:1, date: DateTime.new(2017,9,4,8,05,0,'+0300'), value:10}, 
    {id:4, item:1, date: DateTime.new(2017,9,4,18,19,10,'+0300'), value:1}, 
    {id:5, item:2, date: DateTime.new(2017,9,4,19,20,0,'+0300'), value:2}, 
    {id:6, item:2, date: DateTime.new(2017,9,4,22,22,0,'+0300'), value:5}, 
    {id:7, item:2, date: DateTime.new(2017,9,5,23,0,0,'+0300'), value:1}] 

J'ai besoin nicher sous chaque h1 tout hachage de h2, qui correspondent à des critères:

  1. item valeur (par exemple, item:1 dans les deux hachages)
  2. date de h2 est en from - to ge dans h1

Jusqu'à présent, je peux tout simplement nid tous hash sous h1 avec celui-ci:

my_hash = h1.each do 
    |mh| mh[:inventory]=h2 
end 

Je crois que les critères chose correspondant pourrait être fait avec ceci:

h2.find{ |i| i[:item] == h1[:item] && i[:date].between?(h2[:from], h2[:to]) } 

Comment est-ce que je le mets ensemble pour le faire fonctionner, s'il vous plait? Je vous remercie!

Mise à jour 2

J'essaie de trouver pour chaque h1 appariés avec h2 hash celui-ci:

my_hash = h1.each do |hsh| 
    h2.each do |hsh2| 
    hash=hsh2.find{|h| h[:item] == hsh[:item] && h[:date].between?(hsh[:from],hsh[:to])} if hash 
    hsh[:inventory] = hash 
    end 
end 

mais j'obtiens l'erreur ArgumentError: wrong number of arguments (given 1, expected 0) from (pry):12:in "hash". Qu'est-ce que je fais mal ici, s'il vous plaît?

Mise à jour 3 Si possible sortie idéale serait quelque chose comme ceci:

=> [:order=>{:id=>1, 
    :item=>1, 
    :from=>Mon, 04 Sep 2017 06:00:00 +0300, 
    :to=>Mon, 04 Sep 2017 17:59:59 +0300, 
    :value=>10, 
    :inventory=> 
    {:id=>1, :item=>1, :date=>Mon, 04 Sep 2017 06:10:00 +0300, :value=>5}, 
    {:id=>2, :item=>1, :date=>Mon, 04 Sep 2017 07:20:00 +0300, :value=>7}, 
    {:id=>3, :item=>1, :date=>Mon, 04 Sep 2017 08:05:00 +0300, :value=>10} 
    :order=>{:id=>1, 
    :item=>2, 
    :from=>Mon, 04 Sep 2017 18:00:00 +0300, 
    :to=>Mon, 04 Sep 2017 23:59:59 +0300, 
    :value=>10, 
    :inventory=> 
    {:id=>4, :item=>2, :date=>Mon, 04 Sep 2017 18:19:10 +0300, :value=>1}, 
    {:id=>5, :item=>2, :date=>Mon, 04 Sep 2017 19:20:00 +0300, :value=>2}, 
    {:id=>6, :item=>2, :date=>Mon, 04 Sep 2017 22:22:00 +0300, :value=>5}, 
    {:id=>7, :item=>2, :date=>Mon, 04 Sep 2017 23:00:00 +0300, :value=>1}}] 

Pour chaque h1 Je voudrais donner de nouvelles clés order puis nid h2 appropriés hash sous clé inventory.

+0

Pourriez-vous ajouter une sortie attendue? –

+0

Il n'y a pas de clé ': inventory' dans aucun des hash d'entrée. – mudasobwa

+0

@ SebastiánPalma S'il vous plaît, voir ma mise à jour 3 ci-dessus où j'ai ajouté l'exemple de sortie + étendu ma question concernant les clés de hachage parent et enfant. – matiss

Répondre

2

Cela fonctionne sur Ruby 2.4.2p198 (14/09/2017 révision 59899) [x 64 mingw32]

h1.each do |hsh1| 
    hsh1[:inventory] = 
     h2.find do |hsh2| 
      hsh2[:item] == hsh1[:item] && 
      hsh2[:date].between?(hsh1[:from],hsh1[:to]) 
     end || Hash.new 
end 

Vous pouvez jouer avec la valeur par défaut de HSH1: en remplaçant Hash.new avec [inventaire] la valeur par défaut désirée.

+0

Merci, cependant il semble niché seulement un seul hash de 'h2' sous' h1', pas tout possible. Pourriez-vous jeter un coup d'oeil sur ma mise à jour 3 ci-dessus où j'ai ajouté un exemple de sortie et effacé ma question, s'il vous plaît? – matiss

+0

Silly moi, je peux substituer 'find' avec' find_all' et alors il va imbriquer tous les hashes correspondant à mes critères :) – matiss