Reclassing an instance in Python

前端 未结 8 555
野趣味
野趣味 2020-12-02 10:07

I have a class that is provided to me by an external library. I have created a subclass of this class. I also have an instance of the original class.

I now want to t

相关标签:
8条回答
  • 2020-12-02 10:09

    I'm not sure that the use of inheritance is best in this case (at least with regards to "reclassing"). It seems like you're on the right track, but it sounds like composition or aggregation would be best for this. Here's an example of what I'm thinking of (in untested, pseudo-esque code):

    from copy import copy
    
    # As long as none of these attributes are defined in the base class,
    # this should be safe
    class SkilledProgrammer(Programmer):
        def __init__(self, *skillsets):
            super(SkilledProgrammer, self).__init__()
            self.skillsets = set(skillsets)
    
    def teach(programmer, other_programmer):
        """If other_programmer has skillsets, append this programmer's
           skillsets.  Otherwise, create a new skillset that is a copy
           of this programmer's"""
        if hasattr(other_programmer, skillsets) and other_programmer.skillsets:
            other_programmer.skillsets.union(programmer.skillsets)
        else:
            other_programmer.skillsets = copy(programmer.skillsets)
    def has_skill(programmer, skill):
        for skillset in programmer.skillsets:
            if skill in skillset.skills
                return True
        return False
    def has_skillset(programmer, skillset):
        return skillset in programmer.skillsets
    
    
    class SkillSet(object):
        def __init__(self, *skills):
            self.skills = set(skills)
    
    C = SkillSet("malloc","free","pointer arithmetic","curly braces")
    SQL = SkillSet("SELECT", "INSERT", "DELETE", "UPDATE")
    
    Bob = SkilledProgrammer(C)
    Jill = Programmer()
    
    teach(Bob, Jill)          #teaches Jill C
    has_skill(Jill, "malloc") #should return True
    has_skillset(Jill, SQL)   #should return False
    

    You may have to read more about sets and arbitrary argument lists if you aren't familiar with them to get this example.

    0 讨论(0)
  • 2020-12-02 10:09

    This is fine. I've used this idiom plenty of times. One thing to keep in mind though is that this idea doesn't play well with old-style classes and various C extensions. Normally this wouldn't be an issue, but since you are using an external library you'll just have to make sure you're not dealing with any old-style classes or C extensions.

    0 讨论(0)
  • 2020-12-02 10:14

    I will say this is perfectly fine, if it works for you.

    0 讨论(0)
  • 2020-12-02 10:21

    "The State Pattern allows an object to alter its behavior when its internal state changes. The object will appear to change it's class." - Head First Design Pattern. Something very similar write Gamma et.al. in their Design Patterns book. (I have it at my other place, so no quote). I think that's the whole point of this design pattern. But if I can change the class of an object at runtime, most of the time i don't need the pattern (there are cases when State Pattern does more than simulate a class change).

    Also, changing class at runtime doesn't always work:

    class A(object):
        def __init__(self, val):
            self.val = val
        def get_val(self):
            return self.val
    
    class B(A):
        def __init__(self, val1, val2):
            A.__init__(self, val1)
            self.val2 = val2
        def get_val(self):
            return self.val + self.val2
    
    
    a = A(3)
    b = B(4, 6)
    
    print a.get_val()
    print b.get_val()
    
    a.__class__ = B
    
    print a.get_val() # oops!
    

    Apart from that, I consider changing class at runtime Pythonic and use it from time to time.

    0 讨论(0)
  • 2020-12-02 10:23

    This technique seems reasonably Pythonic to me. Composition would also be a good choice, but assigning to __class__ is perfectly valid (see here for a recipe that uses it in a slightly different way).

    0 讨论(0)
  • 2020-12-02 10:24

    Reclassing instances like this is done in Mercurial (a distributed revision control system) when extensions (plugins) want to change the object that represent the local repository. The object is called repo and is initially a localrepo instance. It is passed to each extension in turn and, when needed, extensions will define a new class which is a subclass of repo.__class__ and change the class of repo to this new subclass!

    It looks like this in code:

    def reposetup(ui, repo):
        # ...
    
        class bookmark_repo(repo.__class__): 
            def rollback(self):
                if os.path.exists(self.join('undo.bookmarks')):
                    util.rename(self.join('undo.bookmarks'), self.join('bookmarks'))
                return super(bookmark_repo, self).rollback() 
    
            # ...
    
        repo.__class__ = bookmark_repo 
    

    The extension (I took the code from the bookmarks extension) defines a module level function called reposetup. Mercurial will call this when initializing the extension and pass a ui (user interface) and repo (repository) argument.

    The function then defines a subclass of whatever class repo happens to be. It would not suffice to simply subclass localrepo since extensions need to be able to extend each other. So if the first extension changes repo.__class__ to foo_repo, the next extension should change repo.__class__ to a subclass of foo_repo and not just a subclass of localrepo. Finally the function changes the instanceø's class, just like you did in your code.

    I hope this code can show a legitimate use of this language feature. I think it's the only place where I've seen it used in the wild.

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