Efficient implementation of immutable (double) LinkedList

前端 未结 3 1296
难免孤独
难免孤独 2020-11-30 05:12

Having read this question Immutable or not immutable? and reading answers to my previous questions on immutability, I am still a bit puzzled about efficient implementation o

相关标签:
3条回答
  • 2020-11-30 05:39

    Should we recursively copy all the references to the list when we insert an element?

    You should recursively copy the prefix of the list up until the insertion point, yes.

    That means that insertion into an immutable linked list is O(n). (As is inserting (not overwriting) an element in array).

    For this reason insertion is usually frowned upon (along with appending and concatenation).

    The usual operation on immutable linked lists is "cons", i.e. appending an element at the start, which is O(1).

    You can see clearly the complexity in e.g. a Haskell implementation. Given a linked list defined as a recursive type:

    data List a = Empty | Node a (List a)
    

    we can define "cons" (inserting an element at the front) directly as:

    cons a xs = Node a xs
    

    Clearly an O(1) operation. While insertion must be defined recursively -- by finding the insertion point. Breaking the list into a prefix (copied), and sharing that with the new node and a reference to the (immutable) tail.

    The important thing to remember about linked lists is :

    • linear access

    For immutable lists this means:

    • copying the prefix of a list
    • sharing the tail.

    If you are frequently inserting new elements, a log-based structure , such as a tree, is preferred.

    0 讨论(0)
  • 2020-11-30 05:45

    Supposedly we have a general class of Node and class LinkedList based on the above allowing the user to add, remove etc. Now, how would we ensure immutability?

    You ensure immutability by making every field of the object readonly, and ensuring that every object referred to by one of those readonly fields is also an immutable object. If the fields are all readonly and only refer to other immutable data, then clearly the object will be immutable!

    Should we recursively copy all the references to the list when we insert an element?

    You could. The distinction you are getting at here is the difference between immutable and persistent. An immutable data structure cannot be changed. A persistent data structure takes advantage of the fact that a data structure is immutable in order to re-use its parts.

    A persistent immutable linked list is particularly easy:

    abstract class ImmutableList
    {
        public static readonly ImmutableList Empty = new EmptyList();
        private ImmutableList() {}
        public abstract int Head { get; }
        public abstract ImmutableList Tail { get; }
        public abstract bool IsEmpty { get; }
        public abstract ImmutableList Add(int head);
        private sealed class EmptyList : ImmutableList
        {
            public override int Head { get {  throw new Exception(); } }
            public override ImmutableList Tail { get { throw new Exception(); } }
            public override bool IsEmpty { get { return true; } }
            public override ImmutableList Add(int head)
            {
                return new List(head, this);
            }
        }
    
        private sealed class List : ImmutableList
        {
            private readonly int head;
            private readonly ImmutableList tail;
            public override int Head { get { return head; } }
            public override ImmutableList Tail { get { return tail; } }
            public override bool IsEmpty { get { return false; } }
            public override ImmutableList Add(int head)
            {
                return new List(head, this);
            }
        }
    }
    ...
    ImmutableList list1 = ImmutableList.Empty;
    ImmutableList list2 = list1.Add(100);
    ImmutableList list3 = list2.Add(400);
    

    And there you go. Of course you would want to add better exception handling and more methods, like IEnumerable<int> methods. But there is a persistent immutable list. Every time you make a new list, you re-use the contents of an existing immutable list; list3 re-uses the contents of list2, which it can do safely because list2 is never going to change.

    Would any of your answers also work on doubly-linked lists?

    You can of course easily make a doubly-linked list that does a full copy of the entire data structure every time, but that would be dumb; you might as well just use an array and copy the entire array.

    Making a persistent doubly-linked list is quite difficult but there are ways to do it. What I would do is approach the problem from the other direction. Rather than saying "can I make a persistent doubly-linked list?" ask yourself "what are the properties of a doubly-linked list that I find attractive?" List those properties and then see if you can come up with a persistent data structure that has those properties.

    For example, if the property you like is that doubly-linked lists can be cheaply extended from either end, cheaply broken in half into two lists, and two lists can be cheaply concatenated together, then the persistent structure you want is an immutable catenable deque, not a doubly-linked list. I give an example of a immutable non-catenable deque here:

    http://blogs.msdn.com/b/ericlippert/archive/2008/02/12/immutability-in-c-part-eleven-a-working-double-ended-queue.aspx

    Extending it to be a catenable deque is left as an exercise; the paper I link to on finger trees is a good one to read.

    UPDATE:

    according to the above we need to copy prefix up to the insertion point. By logic of immutability, if w delete anything from the prefix, we get a new list as well as in the suffix... Why to copy only prefix then, and not suffix?

    Well consider an example. What if we have the list (10, 20, 30, 40), and we want to insert 25 at position 2? So we want (10, 20, 25, 30, 40).

    What parts can we reuse? The tails we have in hand are (20, 30, 40), (30, 40) and (40). Clearly we can re-use (30, 40).

    Drawing a diagram might help. We have:

    10 ----> 20 ----> 30 -----> 40 -----> Empty
    

    and we want

    10 ----> 20 ----> 25 -----> 30 -----> 40 -----> Empty
    

    so let's make

    | 10 ----> 20 --------------> 30 -----> 40 -----> Empty
    |                        /
    | 10 ----> 20 ----> 25 -/ 
    

    We can re-use (30, 40) because that part is in common to both lists.

    UPDATE:

    Would it be possible to provide the code for random insertion and deletion as well?

    Here's a recursive solution:

    ImmutableList InsertAt(int value, int position)
    {
        if (position < 0) 
            throw new Exception();
        else if (position == 0) 
             return this.Add(value);
        else 
            return tail.InsertAt(value, position - 1).Add(head);
    }
    

    Do you see why this works?

    Now as an exercise, write a recursive DeleteAt.

    Now, as an exercise, write a non-recursive InsertAt and DeleteAt. Remember, you have an immutable linked list at your disposal, so you can use one in your iterative solution!

    0 讨论(0)
  • 2020-11-30 05:51

    There is a way to emulate "mutation" : using immutable maps.

    For a linked list of Strings (in Scala style pseudocode):

    case class ListItem(s:String, id:UUID, nextID: UUID)

    then the ListItems can be stored in a map where the key is UUID: type MyList = Map[UUID, ListItem]

    If I want to insert a new ListItem into val list : MyList :

    def insertAfter(l:MyList, e:ListItem)={ 
         val beforeE=l.getElementBefore(e)
         val afterE=l.getElementAfter(e)
         val eToInsert=e.copy(nextID=afterE.nextID)
         val beforeE_new=beforeE.copy(nextID=e.nextID)
         val l_tmp=l.update(beforeE.id,beforeE_new)
         return l_tmp.add(eToInsert)
    }
    

    Where add, update, get takes constant time using Map: http://docs.scala-lang.org/overviews/collections/performance-characteristics

    Implementing double linked list goes similarly.

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