Convert numpy type to python

32,022

Solution 1

It looks like you're correct:

>>> import numpy
>>> import json
>>> json.dumps(numpy.int32(685))
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "/usr/lib/python2.7/json/__init__.py", line 243, in dumps
    return _default_encoder.encode(obj)
  File "/usr/lib/python2.7/json/encoder.py", line 207, in encode
    chunks = self.iterencode(o, _one_shot=True)
  File "/usr/lib/python2.7/json/encoder.py", line 270, in iterencode
    return _iterencode(o, 0)
  File "/usr/lib/python2.7/json/encoder.py", line 184, in default
    raise TypeError(repr(o) + " is not JSON serializable")
TypeError: 685 is not JSON serializable

The unfortunate thing here is that numpy numbers' __repr__ doesn't give you any hint about what type they are. They're running around masquerading as ints when they aren't (gasp). Ultimately, it looks like json is telling you that an int isn't serializable, but really, it's telling you that this particular np.int32 (or whatever type you actually have) isn't serializable. (No real surprise there -- No np.int32 is serializable). This is also why the dict that you inevitably printed before passing it to json.dumps looks like it just has integers in it as well.

The easiest workaround here is probably to write your own serializer1:

class MyEncoder(json.JSONEncoder):
    def default(self, obj):
        if isinstance(obj, numpy.integer):
            return int(obj)
        elif isinstance(obj, numpy.floating):
            return float(obj)
        elif isinstance(obj, numpy.ndarray):
            return obj.tolist()
        else:
            return super(MyEncoder, self).default(obj)

You use it like this:

json.dumps(numpy.float32(1.2), cls=MyEncoder)
json.dumps(numpy.arange(12), cls=MyEncoder)
json.dumps({'a': numpy.int32(42)}, cls=MyEncoder)

etc.

1Or you could just write the default function and pass that as the defaut keyword argument to json.dumps. In this scenario, you'd replace the last line with raise TypeError, but ... meh. The class is more extensible :-)

Solution 2

You could also convert the array to a python list (use the tolist method) and then convert the list to json.

Solution 3

You can use our fork of ujson to deal with NumPy int64. caiyunapp/ultrajson: Ultra fast JSON decoder and encoder written in C with Python bindings and NumPy bindings

pip install nujson

Then

>>> import numpy as np
>>> import nujson as ujson
>>> a = {"a": np.int64(100)}
>>> ujson.dumps(a)
'{"a":100}'
>>> a["b"] = np.float64(10.9)
>>> ujson.dumps(a)
'{"a":100,"b":10.9}'
>>> a["c"] = np.str_("12")
>>> ujson.dumps(a)
'{"a":100,"b":10.9,"c":"12"}'
>>> a["d"] = np.array(list(range(10)))
>>> ujson.dumps(a)
'{"a":100,"b":10.9,"c":"12","d":[0,1,2,3,4,5,6,7,8,9]}'
>>> a["e"] = np.repeat(3.9, 4)
>>> ujson.dumps(a)
'{"a":100,"b":10.9,"c":"12","d":[0,1,2,3,4,5,6,7,8,9],"e":[3.9,3.9,3.9,3.9]}'

Solution 4

If you leave the data in any of the pandas objects, the library supplies a to_json function on Series, DataFrame, and all of the other higher dimension cousins.

See Series.to_json()

Solution 5

In some cases simple json.dump(eval(str(a)), your_file) helps.

Share:
32,022

Related videos on Youtube

ubh
Author by

ubh

Updated on July 09, 2022

Comments

  • ubh
    ubh almost 2 years

    I have a list of dicts in the following form that I generate from pandas. I want to convert it to a json format.

    list_val = [{1.0: 685}, {2.0: 8}]
    output = json.dumps(list_val)
    

    However, json.dumps throws an error: TypeError: 685 is not JSON serializable

    I am guessing it's a type conversion issue from numpy to python(?).

    However, when I convert the values v of each dict in the array using np.int32(v) it still throws the error.

    EDIT: Here's the full code

                new = df[df[label] == label_new] 
                ks_dict = json.loads(content)
                ks_list = ks_dict['variables']
                freq_counts = []
    
                for ks_var in ks_list:
    
                        freq_var = dict()
                        freq_var["name"] = ks_var["name"]
                        ks_series = new[ks_var["name"]]
                        temp_df = ks_series.value_counts().to_dict()
                        freq_var["new"] = [{u: np.int32(v)} for (u, v) in temp_df.iteritems()]            
                        freq_counts.append(freq_var)
    
               out = json.dumps(freq_counts)
    
    • MattDMo
      MattDMo over 9 years
      your code works fine for me... (Python 3.4.2) - [{"1.0": 685}, {"2.0": 8}]
    • Padraic Cunningham
      Padraic Cunningham over 9 years
      so list_val is a numpy array?
    • ubh
      ubh over 9 years
      Yes, it's generated from a DataFrame. I'll update the full code in the post
    • abarnert
      abarnert over 9 years
      So… is there a reason you're putting np.int32(v) instead of v (or int(v); not sure what v is) in freq_var?
    • abarnert
      abarnert over 9 years
      Also, when you have problems like this in the future, try looking at first the repr and then the type of each object, not just printing out their str. (And include the results in your question.) It's a lot easier to just know you have an np.float32 or whatever than to have to guess that maybe there's some kind of type conversion issue.
    • ubh
      ubh over 9 years
      I used int32 assuming that would resolve the Type Error. However, when I changed np.int32(v) to np.int(v),it worked.
  • abarnert
    abarnert over 9 years
    For real fun, try this with np.float64 or np.bool and everything works fine, because they're actually subclasses of float and bool. Once you think about it, it makes sense why those two types are subclasses but none of the other numeric types are, but until you do, it can make for some real fun debugging…
  • mgilson
    mgilson over 9 years
    @abarnert -- np.float64 is obvious (after all, it's just C's double which is what python uses for float), but np.bool isn't so much. It could have been a subclass of np.int32 I would think... Looking at the __mro__ of np.int64, I would expect that one to work too -- at least on python2.x :-).
  • Mack
    Mack about 8 years
    Or you could convert the numpy type to a native type. See stackoverflow.com/a/11389998/2486302 for details on how to do this.
  • nivniv
    nivniv almost 7 years
    This wouldn't work as Series.to_json() still can't handle numpy.ndarrays
  • vk1011
    vk1011 over 6 years
    After a lot of head-banging and resistance to create a custom function or class for a seemingly straightforward problem, this worked for me!! I like it because it keeps things simple!
  • Mark
    Mark over 6 years
    Or you can use the json_tricks library which does this by default (disclaimer: I'm the main contributor).
  • Trevor Boyd Smith
    Trevor Boyd Smith almost 3 years
    unfortunately the json spec does not support complex values... so i think any code that converts complex values to json will be kind of hacky. or at least my attempt smelled hacky.
  • starbeamrainbowlabs
    starbeamrainbowlabs over 2 years
    That's a horrible hack, but it's considerably more concise for a complex data structure I encountered with some float32 instances hidden around.