Project 1B: ArrayDeque61B
Due: Friday, September 20th, 11:59 PM PT
This spec is under construction. It is subject to change until this warning disappears.
FAQ
Each assignment will have an FAQ linked at the top. You can also access it by adding “/faq” to the end of the URL. The FAQ for Project 1B is located here.
Note that this project has limited submission tokens. Please see Submit to the Autograder for more details.
Introduction
In Project 1A, we built LinkedListDeque61B
. Now we’ll see a different
implementation of the Deque61B
interface that uses a backing array, rather
than linked nodes.
By the end of Project 1B, you will…
- Gain an understanding of the implementation of a backing array in data structures.
- Have more experience using testing and test-driven development to verify the correctness of these data structures.
Check out the Project 1B slides for some additional visually oriented tips.
Check out the Getting Started Video for overview of spec.
We will provide relatively little scaffolding. In other words, we’ll say what you should do, but not how.
This section assumes you have watched and fully digested the lectures up till the
ArrayList
lecture, Lecture 7.
For this project, you must work alone! Please carefully read the Policy on Collaboration and Cheating to see what this means exactly. In particular, do not look for solutions online.
It should (still) go without saying that you may not use any of the built-in
java.util
data structures in your implementation! The whole point is to build your own versions! There are a few places where you may use specific data structures outside of tests, and we will clearly say where.
Style
As in Project 1A, we will be enforcing style. You must follow the style guide, or you will be penalized on the autograder.
You can and should check your style locally with the CS 61B plugin. We will not remove the velocity limit for failing to check style.
Getting the Skeleton Files
Follow the instructions in the
Assignment Workflow guide
to get the skeleton code and open it in IntelliJ. For this project, we will be
working in the proj1b
directory.
You see a proj1b
directory appear in your repo with the following structure:
proj1b
├── src
│ └── Deque61B.java
└── tests
└── ArrayDeque61BTest.java
If you get some sort of error, STOP and either figure it out by carefully
reading the git WTFs or seek help at OH
or Ed. You’ll potentially save yourself a lot of trouble vs. guess-and-check
with git commands. If you find yourself trying to use commands recommended by
Google like force push
,
don’t.
Don’t use force push, even if a post you found on Stack Overflow says to do it!
You can also watch Professor Hug’s demo about how to get started and this video if you encounter some git issues.
Deque: ADT and API
If you need a refresher on Deque61B
s, refer to the
Project 1A spec
and the Deque61B.java
file.
Creating the File
Start by creating a file called ArrayDeque61B
. This file should be created
in the proj1b/src
directory. To do this, right-click on the src
directory,
navigate to “New -> Java Class”, and give it the name ArrayDeque61B
.
Just like you did in Project 1A We want our ArrayDeque61B
to be able to hold several different types. To enable this, you should edit the declaration of your class so that it reads:
public class ArrayDeque61B<T>
Recall from lecture that it doesn’t actually matter if we use T
or some other
string like ArrayDeque61B<Glerp>
. However, we recommend using <T>
for
consistency with other Java code.
We also want to tell Java that every ArrayDeque61B
is a Deque61B
, so that users can write code like Deque61B<String> lld1 = new ArrayDeque61B<>();
. To enable this, change the declaration of your class so that it reads:
public class ArrayDeque61B<T> implements Deque61B<T>
Once you’ve done this step, you’ll likely see a squiggly red line under the entire class declaration. This is because you said that your class implements an interface, but you haven’t actually implemented any of the interface methods yet.
Hover over the red line with your mouse, and when the IntelliJ pop-up appears, click the “Implement methods” button. Ensure that all the methods in the list are highlighted, and click “OK”. Now, your class should be filled with a bunch of empty method declarations. These are the methods that you’ll need to implement for this project!
Lastly, you should create an empty constructor. To do this, add the following code to your file, leaving the constructor blank for now.
public ArrayDeque61B() {
}
Note: You can also generate the constructor by clicking “Code”, then “Generate” then “Constructor”, though I prefer the typing the code yourself approach.
Now you’re ready to get started!
ArrayDeque61B
As your second deque implementation, you’ll build the ArrayDeque61B
class. This
deque must use a Java array as the backing data structure.
You may add any private helper classes or methods in ArrayDeque61B.java
if you
deem it necessary.
Constructor
You will need to somehow keep track of what array indices hold the deque’s
front and back elements. We strongly recommend that you treat your array as
circular for this exercise. In other words, if your front item is at position
0
, and you addFirst
, the new front should loop back around to the end of
the array (so the new front item in the deque will be the last item in the
underlying array). This will result in far fewer headaches than non-circular
approaches.
See the Project 1B demo slides for more details. In particular, note that while the conceptual deque and the array contain the same elements, they do not contain them in the same order.
We recommend using the floorMod(int a, int b)
method from Java’s built-in Math
class to assist you in
designing a circular approach. Whereas a % b
might return negative numbers when a is negative, floorMod(int a, int b)
always return non-negative numbers. In practice, this means that the output will have the same sign as the divisor. Here are a few examples
using the floorMod(int a, int b)
method:
int value1 = Math.floorMod(16, 16); // value1 == 0
int value2 = Math.floorMod(-1, 16); // value2 == 15
int value3 = Math.floorMod(20, 16); // value3 == 4
You can use the floorMod(int a, int b)
method by adding the following import statement to the top of your file:
import java.lang.Math;
.
You cannot create an array of generics (e.g. new T[1000]
) in Java for reasons beyond the scope of this course. You will instead need to use the syntax (T[]) new Object[1000]
.
Declare the necessary instance variables, and implement the constructor.
The starting size of your backing array must be
8
.
addFirst
and addLast
As before, implement addFirst
and addLast
. These two methods must not
use looping or recursion. A single add operation must take “constant time,”
that is, adding an element should take approximately the same amount of time no
matter how large the deque is (with one exception). This means that you cannot
use loops that iterate through all / most elements of the deque.
get
Unlike in LinkedListDeque61B
, this method must take constant time.
As before, get
should return null
when the index is invalid (too large or
negative). You should disregard the skeleton code comments for Deque61B.java
for this case.
After you’ve written tests and verified that they fail, implement
get
.
isEmpty
and size
These two methods must take constant time. That is, the time it takes to for either method to finish execution should not depend on how many elements are in the deque.
Write tests for the
isEmpty
andsize
methods, and check that they fail. Then, implement the methods.
toList
toList
will continue to be useful to test your Deque61B
.
Write the toList
method. The first line of the method should be something
like List<T> returnList = new ArrayList<>()
. This is one location where you
are allowed to use a Java data structure.
Some later methods might seem easy if you use
toList
. You may not calltoList
insideArrayDeque61B
; there is a test that checks for this.
Hint One of the other methods may be helpful for implementing this method.
Implement
toList
. You are not given tests this time, so you will need to write them!
All that’s left is to test and implement all the remaining methods. For the rest of this project, we’ll describe our suggested steps at a high level. We strongly encourage you to follow the remaining steps in the order given. In particular, write tests before you implement the method’s functionality. This is called “test-driven development,” and helps ensure that you know what your methods are supposed to do before you do them.
removeFirst
and removeLast
Lastly, write some tests that test the behavior of removeFirst
and
removeLast
, and again ensure that the tests fail.
Do not maintain references to items that are no longer in the deque.
removeFirst
and removeLast
may not use looping or recursion. Like addFirst
and addLast
,
these operations must take "constant time." Refer to the section on writing addFirst
and addLast
for more information on what this means.
getRecursive
Although we are not using a linked list anymore for this project, it is still required to implement this method to keep consistent with our interface. This method technically shouldn’t be in the interface, but it’s here to make testing nice. You can just use this code block for it:
@Override
public T getRecursive(int index) {
throw new UnsupportedOperationException("No need to implement getRecursive for proj 1b");
}
“Implement” getRecursive
.
Resizing
We recommend you complete the other methods first, verify that they are working correctly without resizing, and come back to resizing after.
Resizing Up
The exception to an Array Deque’s “constant time” requirement is when the array fills, and you need to “resize” to have enough space to add the next element. In this case, you can take “linear time” to resize the array before adding the element.
Correctly resizing your array is very tricky, and will require some deep thought. Try drawing out various approaches by hand. It may take you quite some time to come up with the right approach, and we encourage you to debate the big ideas with your fellow students or TAs. Make sure that your actual implementation is by you alone.
Make sure to resize by a geometric factor.
We do not recommend using
arraycopy
with a circular implementation. It will work, but results in a significantly more complex (and harder to debug!) implementation than necessary.Instead, we suggest thinking forward to how you might implement
get
and using afor
loop in some way.
Remember to implement
addFirst
andaddLast
first, and write tests to verify that they are correct. Make sure to add enough elements so that your backing array resizes! For more info on resizing, check out these slides.
Resizing Down
The amount of memory that your program uses at any given time must be proportional to the number of items. For example, if you add 10,000 items to the deque, and then remove 9,999 items, you shouldn’t still be using an array that can hold 10,000 items. For arrays of length 16 or more, your usage factor should always be at least 25%. This means that before performing a remove operation, if the number of elements in the array is at or under 25% the length of the array, you should resize the array down. For arrays length 15 or less, your usage factor can be arbitrarily low.
We, again, do not recommend using
arraycopy
with a circular implementation. If you followed our advice above to use afor
loop to resize up, resizing down should look very similar to resizing up (perhaps a helper method?).
After you’ve written tests and verified that they fail, implement
removeFirst
andremoveLast
.
For the intended experience, follow these steps in order. If you do something else and ask us for help, we will refer you back to these steps.
Writing Tests
Refer to the Project 1A spec for a review of how to write tests. Similar to Project 1A, you will be scored on the coverage of your unit tests for Project 1B. You might find some of your tests from Project 1A to be reusable in this project; don’t be afraid to copy them over!
Suggestions
- Try to get everything working for a fixed-size array first. This would be good point to start to familiarize yourself.
- Once you are confident working solution for a fixed-size array, try resizing - consider having a helper method for it!
- DO NOT modify
Deque61B
interface.
Deque61B
Enhancements
This section requires information covered in Lecture 11. Iterators, Object Methods. Because this lecture occurs after the due date, we are guaranteeing 3-day extensions for this assignment. You can also watch Josh Hug’s videos on the subject or read the textbook.
In this section of the project, you are going to expand upon the functionality of the Deque61B
interface.
Object Methods
In order to implement the following methods, you should start by copying and pasting your Project 1A
implementation of LinkedListDeque61B
into the src
directory.
iterator()
One shortcoming of our Deque61B
interface is that it can not be iterated over. That is, the code below fails to compile with the error “foreach not applicable to type”.
Deque61B<String> lld1 = new LinkedListDeque61B<>();
lld1.addLast("front");
lld1.addLast("middle");
lld1.addLast("back");
for (String s : lld1) {
System.out.println(s);
}
Similarly, if we try to write a test that our Deque61B
contains a specific set of items, we’ll also get a compile error, in this case: “Cannot resolve method containsExactly in Subject”.
public void addLastTestBasicWithoutToList() {
Deque61B<String> lld1 = new LinkedListDeque61B<>();
lld1.addLast("front"); // after this call we expect: ["front"]
lld1.addLast("middle"); // after this call we expect: ["front", "middle"]
lld1.addLast("back"); // after this call we expect: ["front", "middle", "back"]
assertThat(lld1).containsExactly("front", "middle", "back");
}
Again the issue is that our item cannot be iterated over. The Truth
library works by iterating over our object (as in the first example), but our LinkedListDeque61B
does not support iteration.
To fix this, you should first modify the Deque61B
interface so that the declaration reads:
public interface Deque61B<T> extends Iterable<T> {
Next, implement the iterator()
method using the techniques described in lecture 11.
Task: Implement the
iterator()
method in bothLinkedListDeque61B
andArrayDeque61B
according to lecture.
You are not allowed to call toList
here.
equals()
Consider the following code:
@Test
public void testEqualDeques61B() {
Deque61B<String> lld1 = new LinkedListDeque61B<>();
Deque61B<String> lld2 = new LinkedListDeque61B<>();
lld1.addLast("front");
lld1.addLast("middle");
lld1.addLast("back");
lld2.addLast("front");
lld2.addLast("middle");
lld2.addLast("back");
assertThat(lld1).isEqualTo(lld2);
}
If we run this code, we see that we fail the test, with the following message:
expected: [front, middle, back]
but was : (non-equal instance of same class with same string representation)
The issue is that the Truth
library is using the equals
method of the LinkedListDeque61B
class. The default implementation is given by the code below:
public boolean equals(Object obj) {
return (this == obj);
}
That is, the equals method simply checks to see if the addresses of the two objects are the same. We want to be able to check whether the two Deque61B
objects are equal in terms of elements and order so therefore we need a different equals
method.
Override the equals method in the ArrayDeque61B
and LinkedListDeque61B
classes. For guidance on writing an equals
method, see the lecture slides or the lecture code repository.
Note: You might ask why we’re implementing the same method in two classes rather than providing a
default
method in theDeque61B
interface. Interfaces are not allowed to providedefault
methods that overrideObject
methods. For more see https://stackoverflow.com/questions/24595266/why-is-it-not-allowed-add-tostring-to-interface-as-default-method.However, one workaround for this is to provide a
default
, non-Object
helper method in theDeque61B
interface and have the implementing classes call it.
Override the equals()
method in the LinkedListDeque61B
and ArrayDeque61B
classes.
Important: You should not use
getClass
, and there’s no need to do any casting in yourequals
method. That is, you shouldn’t be doing(ArrayDeque61B) o
. Suchequals
methods are old fashioned and overly complex. Useinstanceof
instead.Note: The
instanceof
operator behaves a little strangely with generic types, for reasons beyond the scope of this course. For example, if you want to check iflst
is an instance of aList<Integer>
, you should uselst instanceof List<?>
rather thanlst instanceof List<Integer>
. Unfortunately, this is not able to check the types of the elements, but it’s the best we can do.
Important: Make sure you use the @Override
tag when overriding methods. A common mistake in student code is to try to override equals(ArrayList<T> other)
rather than equals(Object other)
. Using the optional @Override
tag will prevent your code from compiling if you make this mistake. @Override
is a great safety net.
You are not allowed to call toList
here.
toString()
Consider the code below, which prints out a LinkedListDeque61B
.
Deque61B<String> lld1 = new LinkedListDeque61B<>();
lld1.addLast("front");
lld1.addLast("middle");
lld1.addLast("back");
System.out.println(lld1);
This code outputs something like deque.proj1a.LinkedListDeque61B@1a04f701
. This is because the print statement implicitly calls the LinkedListDeque61B
toString
method. Since you didn’t override this method, it uses the default, which is given by the code below (you don’t need to understand how this code works).
public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
In turn the hashCode
method, which you have also not overridden, simply returns the address of the object, which in the example above was 1a04f701
.
Task: Override the toString()
method in the LinkedListDeque61B
and ArrayDeque61B
classes, such that the code above prints out [front, middle, back]
.
Hint: Java’s implementation of the
List
interface has atoString
method.Hint: There is a one line solution (see hint 1).
Hint: Your implementation for
LinkedListDeque61B
andArrayDeque61B
should be exactly the same.
Testing The Object Methods
We haven’t provided you with test files for these three object methods; however, we strongly encourage you to use the
techniques you have learned so far to write your own tests. You can structure these tests however you’d like,
since we won’t be testing them. One possible (and suggested) structure is to create a new file in the tests
directory
called LinkedListDeque61BTest
so that you have a testing file for each implementation.
MaxArrayDeque61B
This section requires information covered in Lecture 10: Subtype Polymorphism, Comparators. Because this lecture is on the same day as the due date, we are guaranteeing 3-day extensions for this assignment. You can also watch Josh Hug’s videos on the subject or read the textbook.
After you’ve fully implemented your ArrayDeque61B
and tested its correctness, you will now build the MaxArrayDeque61B
.
A MaxArrayDeque61B
has all the methods that an ArrayDeque61B
has, but it also has 2 additional methods and a new
constructor:
public MaxArrayDeque61B(Comparator<T> c)
: creates aMaxArrayDeque61B
with the givenComparator
. (You may importjava.util.Comparator
for this.)public T max()
: returns the maximum element in the deque as governed by the previously givenComparator
. If theMaxArrayDeque61B
is empty, simply returnnull
.public T max(Comparator<T> c)
: returns the maximum element in the deque as governed by the parameterComparator c
. If theMaxArrayDeque61B
is empty, simply returnnull
.
The MaxArrayDeque61B
can either tell you the max element in itself by using the
Comparator<T>
given to it in the constructor, or an arbitrary Comparator<T>
that is different from the one given in the constructor.
We do not care about the equals(Object o)
method of this class, so feel free to define it however you think is most
appropriate. We will not test this method.
For testing, you can use Comparator.naturalOrder()
in your own test files. This Comparator
is using naturalOrder().
If your generic type is Integer
, you can create your MaxArrayDeque61B
using the following example:
MaxArrayDeque61B<Integer> m = new MaxArrayDeque61B<Integer>(Comparator.naturalOrder());
If you find yourself starting off by copying your entire
ArrayDeque61B
implementation in aMaxArrayDeque61B
file, then you’re not doing this assignment in the intended manner. This is an exercise in clean code, and redundancy is one our worst enemies when battling complexity! For a hint, re-read the second sentence of this section above.
Task: Fill out the MaxArrayDeque61B.java
file according to the API above.
There are no runtime requirements on these additional methods, we only care about the correctness of your answer.
Sometimes, there might be multiple elements in the MaxArrayDeque61B
that are all equal and hence all the max: in in this
case, you can return any of them and they will be considered correct.
You should write tests for this part as well! You’ll
likely be creating multiple Comparator<T>
classes to test your code:
this is the point! To get practice using Comparator
objects to do something useful (find the maximum element) and to
get practice writing your own
Comparator
classes. You will not be turning in these tests, but we still highly suggest making them for your sake.
Submit to the Autograder
Once you’ve written local tests and passed them, try submitting to the autograder. You may or may not pass everything.
- If you fail any of the coverage tests, it means that there is a case that your local tests did not cover. The autograder test name and the test coverage component will give you hints towards the missing case.
- If you fail a correctness test, this means that there is a case that your local tests did not cover, despite having sufficient coverage for flags. This is expected. Coverage flags are an approximation! They also do not provide describe every single behavior that needs to be tested, nor do they guarantee that you assert everything. Here is a list of them!
- If you fail any of the timing tests, it means that your implementation does not meet the timing constraints described above.
- You will have a token limit of 4 tokens every 24 hours. We will not reinstate tokens for failing to add/commit/push your code, run style, etc.
- You may find messages in the autograder response that look something like this:
WARNING: A terminally deprecated method in java.lang.System has been called
. You can safely ignore any line tagged as aWARNING
.
Scoring
This project, similar to Project 0, is divided into individual components, each of which you must implement completely correctly to receive credit.
- Adding: Correctly implement
addFirst
,addLast
, andtoList
. isEmpty
,size
: Correctly implementisEmpty
andsize
with add methods working.get
: Correctly implementget
.- Removing: Correctly implement
removeFirst
andremoveLast
. - Memory: Correctly implement resizing so that you do not use too much memory.
LinkedListDeque61B
Object Methods: Correctly implementiterator
,equals
, andtoString
inLinkedListDeque61B
.ArrayDeque61B
Object Methods: Correctly implementiterator
,equals
, andtoString
inArrayDeque61B
.MaxArrayDeque61B
Functionality: Ensure yourMaxArrayDeque61B
correctly runs all the methods in theDeque61B
interface.MaxArrayDeque61B
Max: Correctly implementmax
inMaxArrayDeque61B
.- Test Coverage: Write tests to capture a sufficient number of flags.
For the test coverage component, we will run your tests against a staff solution and check how many scenarios and edge cases are tested. You can receive partial credit for this component. Here is a list of them!