Ruby JSON parse changes Hash keys

2020-02-17 06:20发布

Lets say I have this Hash:

{
  :info => [
    {
        :from => "Ryan Bates",
        :message => "sup bra",
        :time => "04:35 AM"
    }
  ]
}

I can call the info array by doing hash[:info].

Now when I turn this into JSON (JSON.generate), and then parse it (JSON.parse), I get this hash:

{
  "info" => [
    {
        "from" => "Ryan Bates",
        "message" => "sup bra",
        "time" => "04:35 AM"
    }
  ]
}

Now if I use hash[:info] it returns nil, but not if I use hash["info"].

Why is this? And is there anyway to fix this incompatibility (besides using string keys from the start)?

6条回答
Animai°情兽
2楼-- · 2020-02-17 06:27
  1. Use ActiveSupport::JSON.decode, it will allow you to swap json parsers easier
  2. Use ActiveSupport::JSON.decode(my_json, symbolize_names: true)

This will recursively symbolize all keys in the hash.

(confirmed on ruby 2.0)

查看更多
劳资没心,怎么记你
3楼-- · 2020-02-17 06:30

It's possible to modify all the keys in a hash to convert them from a string to a symbol:

symbol_hash = Hash[obj.map{ |k,v| [k.to_sym, v] }]

puts symbol_hash[:info]
# => {"from"=>"Ryan Bates", "message"=>"sup bra", "time"=>"04:35 AM"}

Unfortunately that doesn't work for the hash nested inside the array. You can, however, write a little recursive method that converts all hash keys:

def symbolize_keys(obj)
  #puts obj.class # Useful for debugging
  return obj.collect { |a| symbolize_keys(a) } if obj.is_a?(Array)
  return obj unless obj.is_a?(Hash)
  return Hash[obj.map{ |k,v| [k.to_sym, symbolize_keys(v)] }]
end

symbol_hash = symbolize_keys(hash)
puts symbol_hash[:info]
# => {:from=>"Ryan Bates", :message=>"sup bra", :time=>"04:35 AM"}
查看更多
看我几分像从前
4楼-- · 2020-02-17 06:32

I solved my similar issue with calling the with_indifferent_access method on it

Here I have a json string and we can assign it to variable s

s = "{\"foo\":{\"bar\":\"cool\"}}"

So now I can parse the data with the JSON class and assign it to h

h = JSON.parse(s).with_indifferent_access

This will produce a hash that can accept a string or a symbol as the key

h[:foo]["bar"]
  #=> "cool"
查看更多
ら.Afraid
5楼-- · 2020-02-17 06:41

In short, no. Think about it this way, storing symbols in JSON is the same as storing strings in JSON. So you cannot possibly distinguish between the two when it comes to parsing the JSON string. You can of course convert the string keys back into symbols, or in fact even build a class to interact with JSON which does this automagically, but I would recommend just using strings.

But, just for the sake of it, here are the answers to this question the previous times it's been asked:

what is the best way to convert a json formatted key value pair to ruby hash with symbol as key?

ActiveSupport::JSON decode hash losing symbols

Or perhaps a HashWithIndifferentAccess

查看更多
何必那么认真
6楼-- · 2020-02-17 06:50

The JSON generator converts symbols to strings because JSON does not support symbols. Since JSON keys are all strings, parsing a JSON document will produce a Ruby hash with string keys by default.

You can tell the parser to use symbols instead of strings by using the symbolize_names option.

Example:

original_hash = {:info => [{:from => "Ryan Bates", :message => "sup bra", :time => "04:35 AM"}]}
serialized = JSON.generate(original_hash)
new_hash = JSON.parse(serialized, {:symbolize_names => true})

new_hash[:info]
 #=> [{:from=>"Ryan Bates", :message=>"sup bra", :time=>"04:35 AM"}]

Reference: http://www.ruby-doc.org/stdlib-1.9.3/libdoc/json/rdoc/JSON.html#method-i-parse

查看更多
Luminary・发光体
7楼-- · 2020-02-17 06:51

You can't use that option like this

ActiveSupport::JSON.decode(str_json, symbolize_names: true)

In Rails 4.1 or later, ActiveSupport::JSON.decode no longer accepts an options hash for MultiJSON. MultiJSON reached its end of life and has been removed.

You can use symbolize_keys to handle it.

ActiveSupport::JSON.decode(str_json).symbolize_keys
查看更多
登录 后发表回答