Linked List – Cyclic or Acyclic?

Another problem in the Programming Interviews Exposed book (see previous post) is to determine whether a Linked List is cyclic or not.

The most obvious way to do this is to iterate over the list, checking whether the next element is one that you’ve seen before. So as you get to an element, you sweep back from head to current.

  1. def isAcyclic1(head):
  2. """
  3. >>> e1=Element(1)
  4. >>> e2=Element(2)
  5. >>> e3=Element(3)
  6. >>> e4=Element(4)
  7. >>> e5=Element(5)
  8. >>> e1.next=e2
  9. >>> e2.next=e3
  10. >>> e3.next=e4
  11. >>> e4.next=e5
  12. >>> isAcyclic1(e1)
  13. True
  14. >>> e5.next=e3
  15. >>> isAcyclic1(e1)
  16. False
  17. >>> e5.next=e4
  18. >>> isAcyclic1(e1)
  19. False
  20. >>> e5.next=e2
  21. >>> isAcyclic1(e1)
  22. False
  23. """
  24. curr=head
  25. while curr!=None:
  26. sweeper=head
  27. while sweeper!=curr:
  28. if curr.next==sweeper:
  29. return False
  30. sweeper=sweeper.next
  31. curr=curr.next
  32. return True
  33. class Element:
  34. def __init__(self,data,next=None):
  35. self.data=data
  36. self.next=next
  37. if __name__ == "__main__":
  38. import doctest
  39. doctest.testmod()
def isAcyclic1(head):
    """ 
    >>> e1=Element(1)
    >>> e2=Element(2)
    >>> e3=Element(3)
    >>> e4=Element(4)
    >>> e5=Element(5)
    >>> e1.next=e2
    >>> e2.next=e3
    >>> e3.next=e4
    >>> e4.next=e5
    >>> isAcyclic1(e1)
    True
    >>> e5.next=e3
    >>> isAcyclic1(e1)
    False
    >>> e5.next=e4
    >>> isAcyclic1(e1)
    False                                                                                                                                                                                                           
    >>> e5.next=e2
    >>> isAcyclic1(e1)
    False
    """
    curr=head
    while curr!=None:
        sweeper=head
        while sweeper!=curr:
            if curr.next==sweeper:
                return False
            sweeper=sweeper.next
        curr=curr.next
    return True

class Element:
    def __init__(self,data,next=None):
        self.data=data
        self.next=next

if __name__ == "__main__":
    import doctest
    doctest.testmod()

This is an O(n squared) solution. There’s a better way, of course. You start a fast pointer, skipping elements, which if acyclic will get to the end quickly. If cyclic, it will eventually trip over the slow pointer. It turns out to be an O(n) solution. Smart.

  1. def isAcyclic(head):
  2. """
  3. >>> e1=Element(1)
  4. >>> e2=Element(2)
  5. >>> e3=Element(3)
  6. >>> e4=Element(4)
  7. >>> e5=Element(5)
  8. >>> e1.next=e2
  9. >>> isAcyclic(e1)
  10. True
  11. >>> e2.next=e3
  12. >>> e3.next=e4
  13. >>> e4.next=e5
  14. >>> isAcyclic(e1)
  15. True
  16. >>> e5.next=e3
  17. >>> isAcyclic(e1)
  18. False
  19. >>> e5.next=e4
  20. >>> isAcyclic(e1)
  21. False
  22. >>> e5.next=e2
  23. >>> isAcyclic(e1)
  24. False
  25. """
  26. slow=head
  27. fast=head
  28. while fast!=None and fast.next!=None:
  29. slow=slow.next
  30. fast=fast.next.next
  31. if (fast==slow):
  32. return False
  33. return True
  34. class Element:
  35. def __init__(self,data,next=None):
  36. self.data=data
  37. self.next=next
  38. if __name__ == "__main__":
  39. import doctest
  40. doctest.testmod()
def isAcyclic(head):

    """ 
    >>> e1=Element(1)
    >>> e2=Element(2)
    >>> e3=Element(3)
    >>> e4=Element(4)
    >>> e5=Element(5)
    >>> e1.next=e2
    >>> isAcyclic(e1)
    True
    >>> e2.next=e3
    >>> e3.next=e4
    >>> e4.next=e5
    >>> isAcyclic(e1)
    True
    >>> e5.next=e3
    >>> isAcyclic(e1)
    False
    >>> e5.next=e4
    >>> isAcyclic(e1)
    False
    >>> e5.next=e2
    >>> isAcyclic(e1)
    False                                                                                                                                                                                                           
    """
    slow=head
    fast=head
    while fast!=None and fast.next!=None:
        slow=slow.next
        fast=fast.next.next
        if (fast==slow):
            return False
    return True

class Element:
    def __init__(self,data,next=None):
        self.data=data
        self.next=next

if __name__ == "__main__":
    import doctest
    doctest.testmod()
This entry was posted in programming and tagged , , . Bookmark the permalink.

3 Responses to Linked List – Cyclic or Acyclic?

  1. Pingback: Rebrained! » Blog Archive » Cyclic Linked List – Python function to find the node at the start of the loop

  2. Jasus says:

    Thanks for the nice and simple exalmpe. You’re just missing one crucial piece of information:comparator should specify a custom comparison function of two arguments (list items) which should return a negative, zero or positive number depending on whether the first argument is considered smaller than, equal to, or larger than the second argument.See note 8 under of the official python documentation.

  3. DanielTooms says:

    ?? ????? ?????? ???????? ??? ?????? ?????? ??? ???? ??????. ??? ??? ???? ??? ??? ???? ????? 21 ???? ?? ????? ????? ?????? ???????. ?? ????? ??? ???? ?? ??? ????, ??? ??? ??? ?? ????? ????? ???. ?? ?????? ?? ????, ??? ????? ?? ????? ?????, ?? ???? ??? ??? ????? ????. ???? ????? ????? ???? ???? ???? ??? ??? ???? ???? ??? ?????? ???? ????? ???????? ???? ???????? ???. ????? ????? ??? ????? ????? ?????? ????. ?? ?? ??? ??? ???? ????? ?? ??? ?? ???? ?????? ????? ???????, ???? ?????? ???? ?????? ???? ???? ?????? ??????????.
    https://israelnightclub.com/?????-?????-?????/
    https://israelnightclub.com/?????-?????-?????/

Leave a Reply

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