Both singly-linked and doubly-linked lists are folklore, having been
used in programs for over 40 years. They are discussed, for example,
by Knuth [41, Sections 2.2.3-2.2.5]. Even the
data
structure seems to be a well-known data structures exercise.
Exercise 3..1
Why is it not possible, in an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
to use a dummy node to avoid
all the special cases that occur in the operations
data:image/s3,"s3://crabby-images/be0f0/be0f081e5d63e562049cb51d9d3ef2b83798091c" alt="$ \mathtt{push(x)}$"
,
data:image/s3,"s3://crabby-images/692b9/692b992ca1f072ec312e6a4ab1a1bf13960e5bcb" alt="$ \mathtt{pop()}$"
,
data:image/s3,"s3://crabby-images/26dc3/26dc351f30373fafdd0cc1756c143825e42c462d" alt="$ \mathtt{add(x)}$"
, and
data:image/s3,"s3://crabby-images/59775/597758e8804408aef1c1798115bf378b93d05dcc" alt="$ \mathtt{remove()}$"
?
Exercise 3..2
Design and implement an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
method,
data:image/s3,"s3://crabby-images/a78db/a78db631116f24a0b8b5caf9c412b128f35a1e82" alt="$ \mathtt{secondLast()}$"
, that returns
the second-last element of an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
. Do this without using the
member variable,
data:image/s3,"s3://crabby-images/8831c/8831c025e0c7d2ed3ddfa5e87085a075d5c855f9" alt="$ \mathtt{n}$"
, that keeps track of the size of the list.
Exercise 3..3
Describe and implement the
data:image/s3,"s3://crabby-images/253b8/253b8969da71c715229c2a9d98a3996819414355" alt="$ \mathtt{List}$"
operations
data:image/s3,"s3://crabby-images/c77cd/c77cd9a03a4ae107dac5a9f38c3c342e3ae4b0e3" alt="$ \mathtt{get(i)}$"
,
data:image/s3,"s3://crabby-images/ed3bf/ed3bf0693e7f9c29d77d5fecc9a7da799a29f76f" alt="$ \mathtt{set(i,x)}$"
,
data:image/s3,"s3://crabby-images/d6cf0/d6cf0a46f67643b01dbd4085d9da622a53e69922" alt="$ \mathtt{add(i,x)}$"
and
data:image/s3,"s3://crabby-images/0bf19/0bf199d3661879586f584045bf093ea89bc8d6c0" alt="$ \mathtt{remove(i)}$"
on an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
. Each of these operations
should run in
data:image/s3,"s3://crabby-images/7129a/7129a69be7dcba02a0b034f71270166bb7c008af" alt="$ O(1+\ensuremath{\mathtt{i}})$"
time.
Exercise 3..4
Design and implement an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
method,
data:image/s3,"s3://crabby-images/bd8a6/bd8a6d85803716199f5e0efc787d7404f3bc3f8d" alt="$ \mathtt{reverse()}$"
that reverses the
order of elements in an
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
. This method should run in
data:image/s3,"s3://crabby-images/0537c/0537c8c77a35c370fbe4161a769dfb9e019105a6" alt="$ O(\ensuremath{\mathtt{n}})$"
time, should not use recursion, should not use any secondary data
structures, and should not create any new nodes.
Exercise 3..5
Design and implement
data:image/s3,"s3://crabby-images/ffdc6/ffdc61db66ebcecf2f856a6b091b32514be1106f" alt="$ \mathtt{SLList}$"
and
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
methods called
data:image/s3,"s3://crabby-images/f5c41/f5c41f3b8af19ddb764c28507a8cf38222132e19" alt="$ \mathtt{checkSize()}$"
.
These methods walk through the list and count the number of nodes to
see if this matches the value,
data:image/s3,"s3://crabby-images/8831c/8831c025e0c7d2ed3ddfa5e87085a075d5c855f9" alt="$ \mathtt{n}$"
, stored in the list. These methods
return nothing, but throw an exception if the size they compute does
not match the value of
data:image/s3,"s3://crabby-images/8831c/8831c025e0c7d2ed3ddfa5e87085a075d5c855f9" alt="$ \mathtt{n}$"
.
Exercise 3..6
Without referring to this chapter, try to recreate the code for the
data:image/s3,"s3://crabby-images/550cd/550cd0e4d8553cce8e1d09630b862bc6b70329d4" alt="$ \mathtt{addBefore(w)}$"
operation, that creates a node,
data:image/s3,"s3://crabby-images/07d79/07d79e9fccb5c80668be13b4c9b6546e6a1ea406" alt="$ \mathtt{u}$"
, and adds it just
before the node
data:image/s3,"s3://crabby-images/36e8d/36e8d62631e5004e67c4d6a449ef39898e681615" alt="$ \mathtt{w}$"
in a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
. If your code does not exactly match
the code given in this book it may still be correct. Test it and see
if it works.
The next few exercises involve performing manipulations on
s.
These should all be done without allocating any new nodes or temporary
arrays. More specifically, they can all be done only by changing
the
and
values of existing nodes.
Exercise 3..7
Write a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
method
data:image/s3,"s3://crabby-images/edad7/edad78c11adfde2ab8de517e8b147f31cda87b54" alt="$ \mathtt{isPalindrome()}$"
that returns
data:image/s3,"s3://crabby-images/4be11/4be1101521747789b1bad4c3fcdc999d84617033" alt="$ \mathtt{true}$"
if the
list is a palindrome, i.e., the element at position
data:image/s3,"s3://crabby-images/ec1c5/ec1c5e79f3915f1862d24865cddb5ac16943f9ba" alt="$ \mathtt{i}$"
is equal to
the element at position
data:image/s3,"s3://crabby-images/539e5/539e567139ddedd0cabe7025c59ea01884986eec" alt="$ \ensuremath{\mathtt{n}}-i-1$"
for all
data:image/s3,"s3://crabby-images/995fa/995faa55a85e060f53e2e37b47d9ab6820906c4d" alt="$ i\in\{0,\ldots,\ensuremath{\mathtt{n}}-1\}$"
.
Your code should run in
data:image/s3,"s3://crabby-images/0537c/0537c8c77a35c370fbe4161a769dfb9e019105a6" alt="$ O(\ensuremath{\mathtt{n}})$"
time.
Exercise 3..8
Implement a method
data:image/s3,"s3://crabby-images/056da/056da6e2c7c28beea7c20d733fb50fac55f7e6f6" alt="$ \mathtt{rotate(r)}$"
that ``rotates'' a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
so that list
item
data:image/s3,"s3://crabby-images/ec1c5/ec1c5e79f3915f1862d24865cddb5ac16943f9ba" alt="$ \mathtt{i}$"
becomes list item
data:image/s3,"s3://crabby-images/255eb/255eb2588b805002a9cc046129449fc779abcbf7" alt="$ (\ensuremath{\mathtt{i}}+\ensuremath{\mathtt{r}})\bmod \ensuremath{\mathtt{n}}$"
. This method should
run in
data:image/s3,"s3://crabby-images/a6f08/a6f08cc9114ae2544c7ecc6cfaf9ea554ae4e872" alt="$ O(1+\min\{\ensuremath{\mathtt{r}},\ensuremath{\mathtt{n}}-\ensuremath{\mathtt{r}}\})$"
time and should not modify any nodes in
the list.
Exercise 3..9
Write a method,
data:image/s3,"s3://crabby-images/bf79c/bf79ce862847b676e8ec919771b85ab79269206d" alt="$ \mathtt{truncate(i)}$"
, that truncates a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
at position
data:image/s3,"s3://crabby-images/ec1c5/ec1c5e79f3915f1862d24865cddb5ac16943f9ba" alt="$ \mathtt{i}$"
. After the execution of this method, the size of the list is
data:image/s3,"s3://crabby-images/ec1c5/ec1c5e79f3915f1862d24865cddb5ac16943f9ba" alt="$ \mathtt{i}$"
and it contains only the elements at indices
data:image/s3,"s3://crabby-images/d4fb0/d4fb0d4a6c4145b2e27ecf9f9882e907db027f87" alt="$ 0,\ldots,\ensuremath{\mathtt{i}}-1$"
. The
return value is another
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
that contains the elements at indices
data:image/s3,"s3://crabby-images/af346/af3462b638941987da5a91740ecee06ad5513ad0" alt="$ \ensuremath{\mathtt{i}},\ldots,\ensuremath{\mathtt{n}}-1$"
. This method should run in
data:image/s3,"s3://crabby-images/bdfb7/bdfb7029d3154adc3f4e5cd21ab6d717ebc28209" alt="$ O(\min\{\ensuremath{\mathtt{i}},\ensuremath{\mathtt{n}}-\ensuremath{\mathtt{i}}\})$"
time.
Exercise 3..10
Write a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
method,
data:image/s3,"s3://crabby-images/0a7c8/0a7c8218b597b61bfb153b5ef0411b0303c6b98c" alt="$ \mathtt{absorb(l2)}$"
, that takes as an argument
a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
,
data:image/s3,"s3://crabby-images/5ac7f/5ac7fd1b9ba2cc72118ae60e9702626baa1cf9f3" alt="$ \mathtt{l2}$"
, empties it and appends its contents, in order,
to the receiver. For example, if
data:image/s3,"s3://crabby-images/d229f/d229fd038166793340709a6a3b771f005d94009e" alt="$ \mathtt{l1}$"
contains
data:image/s3,"s3://crabby-images/4dd13/4dd137b80c04115051c778f437bf58ed8fa7de6e" alt="$ a,b,c$"
and
data:image/s3,"s3://crabby-images/5ac7f/5ac7fd1b9ba2cc72118ae60e9702626baa1cf9f3" alt="$ \mathtt{l2}$"
contains
data:image/s3,"s3://crabby-images/146cf/146cf9f66ed00642d22ecc768eb89e61e6b9837f" alt="$ d,e,f$"
, then after calling
data:image/s3,"s3://crabby-images/0e077/0e0779eaa94a2ceca6292844e27d9f3f344c69f5" alt="$ \mathtt{l1.absorb(l2)}$"
,
data:image/s3,"s3://crabby-images/d229f/d229fd038166793340709a6a3b771f005d94009e" alt="$ \mathtt{l1}$"
will contain
data:image/s3,"s3://crabby-images/99af7/99af7876523e927e9230e0b0d61f1e4147fd9dc1" alt="$ a,b,c,d,e,f$"
and
data:image/s3,"s3://crabby-images/5ac7f/5ac7fd1b9ba2cc72118ae60e9702626baa1cf9f3" alt="$ \mathtt{l2}$"
will be empty.
Exercise 3..11
Write a method
data:image/s3,"s3://crabby-images/61f4d/61f4dade32628eac3089c5f555b73e296f7ac659" alt="$ \mathtt{deal()}$"
that removes all the elements with odd-numbered
indices from a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
and return a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
containing these elements.
For example, if
data:image/s3,"s3://crabby-images/d229f/d229fd038166793340709a6a3b771f005d94009e" alt="$ \mathtt{l1}$"
, contains the elements
data:image/s3,"s3://crabby-images/99af7/99af7876523e927e9230e0b0d61f1e4147fd9dc1" alt="$ a,b,c,d,e,f$"
, then after
calling
data:image/s3,"s3://crabby-images/713d8/713d8a5cfb1ef029c89f1e41a008cd3bbdebb7c0" alt="$ \mathtt{l1.deal()}$"
,
data:image/s3,"s3://crabby-images/d229f/d229fd038166793340709a6a3b771f005d94009e" alt="$ \mathtt{l1}$"
should contain
data:image/s3,"s3://crabby-images/db680/db680c1bfa639492473462da492be57d59d1cfb8" alt="$ a,c,e$"
and a list containing
data:image/s3,"s3://crabby-images/93d3e/93d3e5484dc4061582adb2d42dba98a516ac820f" alt="$ b,d,f$"
should be returned.
Exercise 3..12
Write a method,
data:image/s3,"s3://crabby-images/bd8a6/bd8a6d85803716199f5e0efc787d7404f3bc3f8d" alt="$ \mathtt{reverse()}$"
, that reverses the order of elements in
a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
.
Exercise 3..13
This exercises walks you through an implementation of the merge sort
algorithm for sorting a
data:image/s3,"s3://crabby-images/0f8ca/0f8ca216c7d1ce6ad9a83e9e536a3ebc3c3f6ea7" alt="$ \mathtt{DLList}$"
, as discussed in Section
11.1.1.
- Write a
method called
.
This method takes the first node from
and appends it to the the
receiving list. This is equivalent to
,
except that it should not create a new node.
- Write a
static method,
, that takes two
sorted lists
and
, merges them, and returns a new sorted
list containing the result. This causes
and
to be emptied
in the proces. For example, if
contains
and
contains
, then this method returns a new list containing
.
- Write a
method
that sorts the elements contained
in the list using the merge sort algorithm. This recursive algorithm works as following:
- If the list contains 0 or 1 elements then there is
nothing to do. Otherwise,
- Split the list into two approximately equal length lists
and
using the
method;
- Recursively sort
;
- Recursively sort
; and, finally,
- Merge
and
into a single sorted list.
The next few exercises are more advanced and require a clear
understanding of what happens to the minimum value stored in a
or
as items are added and removed.
Exercise 3..14
Design and implement a
data:image/s3,"s3://crabby-images/da731/da7314d6d5414e9cdd1897e4660c034a6779cb6b" alt="$ \mathtt{MinStack}$"
data structure that can store
comparable elements and supports the stack operations
data:image/s3,"s3://crabby-images/be0f0/be0f081e5d63e562049cb51d9d3ef2b83798091c" alt="$ \mathtt{push(x)}$"
,
data:image/s3,"s3://crabby-images/692b9/692b992ca1f072ec312e6a4ab1a1bf13960e5bcb" alt="$ \mathtt{pop()}$"
, and
data:image/s3,"s3://crabby-images/06fd0/06fd00afb34c0ead58fdf567e3dd4dc8cdec83b7" alt="$ \mathtt{size()}$"
, as well as the
data:image/s3,"s3://crabby-images/672dd/672dd81be5d602de8da12fcda7d940ed873faf5e" alt="$ \mathtt{min()}$"
operation, which
returns the minimum value currently stored in the data structure.
All operations should run in constant time.
Exercise 3..15
Design an implement a
data:image/s3,"s3://crabby-images/32a0a/32a0a07a4ed77d438be027f95a629b80f9d6e4e4" alt="$ \mathtt{MinQueue}$"
data structure that can store
comparable elements and supports the queue operations
data:image/s3,"s3://crabby-images/26dc3/26dc351f30373fafdd0cc1756c143825e42c462d" alt="$ \mathtt{add(x)}$"
,
data:image/s3,"s3://crabby-images/59775/597758e8804408aef1c1798115bf378b93d05dcc" alt="$ \mathtt{remove()}$"
, and
data:image/s3,"s3://crabby-images/06fd0/06fd00afb34c0ead58fdf567e3dd4dc8cdec83b7" alt="$ \mathtt{size()}$"
, as well as the
data:image/s3,"s3://crabby-images/672dd/672dd81be5d602de8da12fcda7d940ed873faf5e" alt="$ \mathtt{min()}$"
operation, which
returns the minimum value currently stored in the data structure.
All operations should run in constant amortized time.
Exercise 3..16
Design an implement a
data:image/s3,"s3://crabby-images/e2cb5/e2cb53e8b8628add759b3a5d24ae500c8da147eb" alt="$ \mathtt{MinDeque}$"
data structure that can store
comparable elements and supports the queue operations
data:image/s3,"s3://crabby-images/1cb7d/1cb7d41f27ac6c58a0a9d06bb0a53786e81afbd1" alt="$ \mathtt{addFirst(x)}$"
,
data:image/s3,"s3://crabby-images/9cf0d/9cf0df645df5becabbc069cc7deedd959917397e" alt="$ \mathtt{removeFirst()}$"
,
data:image/s3,"s3://crabby-images/6e154/6e154d30a78458ba429769ad2bb1c4ef2ba22744" alt="$ \mathtt{removeLast()}$"
and
data:image/s3,"s3://crabby-images/06fd0/06fd00afb34c0ead58fdf567e3dd4dc8cdec83b7" alt="$ \mathtt{size()}$"
, as well
as the
data:image/s3,"s3://crabby-images/672dd/672dd81be5d602de8da12fcda7d940ed873faf5e" alt="$ \mathtt{min()}$"
operation, which returns the minimum value currently
stored in the data structure. All operations should run in constant
amortized time.
The next exercises are designed to test the reader's understanding of
the implementation an analysis of the space-efficient
:
Exercise 3..17
Prove that, if an
data:image/s3,"s3://crabby-images/b70f9/b70f996823fbdc711b935af931d747494acd5157" alt="$ \mathtt{SEList}$"
is used like a
data:image/s3,"s3://crabby-images/61cc2/61cc216e7abfd474a1da105dc16142b45a73be5f" alt="$ \mathtt{Stack}$"
(so that the
modifications are done using
data:image/s3,"s3://crabby-images/3ecf2/3ecf27d76769c3205c7e1ec64691796266c4d1f1" alt="$ \ensuremath{\mathtt{push(x)}}\equiv \ensuremath{\mathtt{add(size(),x)}}$"
and
data:image/s3,"s3://crabby-images/4ec22/4ec2215c9830aac6563eb033b939dc7b26d586f7" alt="$ \ensuremath{\mathtt{pop()}}\equiv \ensuremath{\mathtt{remove(size()-1)}}$"
) then these operations run in
constant amortized time, independent of the value of
data:image/s3,"s3://crabby-images/dea20/dea201779099a60ae023c108097d7a8e850e84e2" alt="$ \mathtt{b}$"
.
Exercise 3..18
Design an implement of a version of an
data:image/s3,"s3://crabby-images/b70f9/b70f996823fbdc711b935af931d747494acd5157" alt="$ \mathtt{SEList}$"
that supports all
the
data:image/s3,"s3://crabby-images/50750/507509639530f08729517396cf28a39934e91900" alt="$ \mathtt{Deque}$"
operations in constant amortized time per operation,
independent of the value of
data:image/s3,"s3://crabby-images/dea20/dea201779099a60ae023c108097d7a8e850e84e2" alt="$ \mathtt{b}$"
.
opendatastructures.org