Deep copy and recursive references

Quick one about coding a deep copy and avoiding recursive references…

Let’s say we have an instance a which has a reference to an instance b and we have to do a deep copy of a. To do this, we need a deep copy of b as well.

But let’s say b had a reference back to a.

Here it is in Python (I know Python has its own __deepcopy__() method but the below is for illustrative purposes).

  1. class A:
  2. def __init__(self,b=None):
  3. self.b=b
  4. def deepcopy(self):
  5. newA=A()
  6. newA.b=self.b.deepcopy()
  7. return newA
  8. class B:
  9. def __init__(self,a=None):
  10. self.a=a
  11. def deepcopy(self):
  12. newB=B()
  13. newB.a=self.a.deepcopy()
  14. return newB
class A:                                                                                                                                                                                                            
    def __init__(self,b=None):
        self.b=b
    def deepcopy(self):
        newA=A()
        newA.b=self.b.deepcopy()
        return newA
class B:
    def __init__(self,a=None):
        self.a=a
    def deepcopy(self):
        newB=B()
        newB.a=self.a.deepcopy()
        return newB

Now let’s execute this:

  • >>> a=A()
  • >>> b=B()
  • >>> a.b=b
  • >>> b.a=a
  • >>> newA=a.deepcopy()
>>> a=A()
>>> b=B()
>>> a.b=b
>>> b.a=a
>>> newA=a.deepcopy()

What happens? Not pretty – a recursion limit error.

To fix this, we can keep track in a as to whether that instance has already created a copy of itself, so when it is called back from b it can just return that instance. This is sometimes called the “memo” copy. See amended code below.

  1. class A:
  2. def __init__(self,b=None):
  3. self.b=b
  4. self.memo=None
  5. def deepcopy(self):
  6. if self.memo==None:
  7. newA=A()
  8. self.memo=newA
  9. newA.b=self.b.deepcopy()
  10. self.memo=None
  11. return newA
  12. else:
  13. return self.memo
  14. class B:
  15. def __init__(self,a=None):
  16. self.a=a
  17. def deepcopy(self):
  18. newB=B()
  19. newB.a=self.a.deepcopy()
  20. return newB
class A:                                                                                                                                                                                                            
    def __init__(self,b=None):
        self.b=b
        self.memo=None
    def deepcopy(self):
        if self.memo==None:
            newA=A()
            self.memo=newA
            newA.b=self.b.deepcopy()
            self.memo=None
            return newA
        else:
            return self.memo
class B:
    def __init__(self,a=None):
        self.a=a
    def deepcopy(self):
        newB=B()
        newB.a=self.a.deepcopy()
        return newB

Now when we execute:

  • >>> a=A()
  • >>> b=B()
  • >>> a.b=b
  • >>> b.a=a
  • >>> newA=a.deepcopy()
  • >>> newA
  • <__main__.A instance at 0x00D53A30>
  • >>> newA.b.a
  • <__main__.A instance at 0x00D53A30>
  • >>> newA2=a.deepcopy()
  • >>> newA2
  • <__main__.A instance at 0x00D53AF8>
  • >>> newA2.b.a
  • <__main__.A instance at 0x00D53AF8>
  • >>>
>>> a=A()
>>> b=B()
>>> a.b=b
>>> b.a=a
>>> newA=a.deepcopy()
>>> newA
<__main__.A instance at 0x00D53A30>
>>> newA.b.a
<__main__.A instance at 0x00D53A30>
>>> newA2=a.deepcopy()
>>> newA2
<__main__.A instance at 0x00D53AF8>
>>> newA2.b.a
<__main__.A instance at 0x00D53AF8>
>>> 

Note that the memo is removed just before the copy is handed back so that if you call deepcopy again you’ll get a new copy of a.

This entry was posted in programming and tagged , , . Bookmark the permalink.

3 Responses to Deep copy and recursive references

  1. Dr. J says:

    It all makes sense written down but this can really mess with your head. Thanks for publishing a very elegant solution.

    Can you provide a link to the reference for “memo” that you hint at above?

  2. Nicolas F. says:

    I couldn’t find any solution. Hopefully you point me to the right direction! Thanks a lot for this example!

  3. nolfonzo says:

    Dr J – I think I first saw the reference here:

    http://docs.python.org/library/copy.html

Leave a Reply

Your email address will not be published. Required fields are marked *