Pickle all attributes except one

后端 未结 6 935
野性不改
野性不改 2021-01-17 07:51

What is the best way to write a __getstate__ method that pickles almost all of an object\'s attributes, but excludes a few?

I have an object wi

相关标签:
6条回答
  • 2021-01-17 07:55

    The only alternative I can think of is some kind of helper function that iterates through an object's properties and adds them (or not) to the dictionary, based on the type.

    Yeah, I think that's pretty much what you're left with, if you want enough "magic" to allow yourself to be lazy (and/or allow for dynamically added attributes). Keep in mind that "pickle can't handle this" isn't the only reason you might not want to include something in the pickled state.

    But it's not as hard as you seem to think, assuming you have code for the "should I pickle this?" logic:

    def __getstate__(self):
      return dict((k, v) for (k, v) in self.__dict__.iteritems() if should_pickle(v))
    
    0 讨论(0)
  • 2021-01-17 07:59

    __slots__ solution

    If you are using slots, you can avoid repeating members to exclude with:

    class C(object):
        _pickle_slots = ['i']
        __slots__ = _pickle_slots + ['j']
        def __init__(self, i, j):
            self.i = i
            self.j = j
        def __getstate__(self):
            return (None, {k:getattr(self, k) for k in C._pickle_slots })
    
    o = pickle.loads(pickle.dumps(C(1, 2), -1))
    
    # i is there
    assert o.i == 1
    
    # j was excluded
    try:
        o.j
    except:
        pass
    else:
        raise
    

    Tested in Python 2.7.6.

    0 讨论(0)
  • 2021-01-17 07:59

    For the your specific case (preventing a function from getting pickled), use this:

    self.__class__.fn = self.__class__.my_func

    Now, instead of adding a function to an instance of a class, you've added it to the class itself, thus the function won't get pickled. This won't work if you want each instance to have its own version of fn.

    My scenario was that I wanted to selectively add get_absolute_url to some Django models, and I wanted to define this in an abstract BaseModel class. I had self.get_absolute_url = … and ran into the pickle issue. Just added __class__ to the assignment solved the issue in my case.

    0 讨论(0)
  • 2021-01-17 08:10

    Using is_instance_method from an earlier answer:

    def __getstate__(self):
        return dict((k, v) for k, v in self.__dict__.iteritems()
                           if not is_instance_method(getattr(self, k)))
    

    Although the is_instance_method operation can also be performed less "magically" by taking an known instance method, say my_func, and taking its type.

    def __getstate__(self):
        instancemethod = type(self.my_func)
        return dict((k, v) for k, v in self.__dict__.iteritems()
                           if not isinstance(getattr(self, k), instancemethod))
    
    0 讨论(0)
  • 2021-01-17 08:10

    You could always just remove the bad items:

    def __getstate__(self):
        state = self.__dict__
        del state[...]
        return state
    
    0 讨论(0)
  • 2021-01-17 08:19

    I'd cut to the root of your problem, and try to serialize the so-called 'un-pickleable' items first. To do this, I'd use dill, which can serialize almost anything in python. Dill also has some good tools for helping you understand what is causing your pickling to fail when your code fails.

    >>> import dill
    >>> dill.loads(dill.dumps(your_bad_object))
    >>> ...
    >>> # if you get a pickling error, use dill's tools to figure out a workaround
    >>> dill.detect.badobjects(your_bad_object, depth=0)
    >>> dill.detect.badobjects(your_bad_object, depth=1)
    >>> ...
    

    If you absolutely wanted to, you could use dill's badobjects (or one of the other detection functions) to dive recursively into your object's reference chain, and pop out the unpickleable objects, instead of calling it at at every depth, as above.

    0 讨论(0)
提交回复
热议问题