Quick link:
Contents:
We recommend that you read through the homework before you begin work to get a sense of the homework overall. It is long but broken into reasonable size chunks.
Beware, this is your first real programming assignment for this class. In previous quarters, students have often found themselves surprised by how much work this assignment involves, and end up using late days. So please start early.
This homework focuses on reading and interpreting specifications, and reading and writing Java code. It also introduces checkRep methods and testing strategies. You will complete the implementation of a graphing polynomial calculator, and you will answer questions about both the code you are given and the code you are going to write.
To complete this homework, you will need to know:
Obtain the starter files by running git pull
.
Commit the solution to problem 0 to your repository in the file hw2/answers.txt before the deadline.
For this problem you will write pseudocode algorithms for arithmetic operations applied to single-variable polynomial equations. An example polynomial is x2 - 2x + 1. Another example is -42 x100 + 22x12 + 5x10. "22x12" is called a "term"; its degree is 12 and its coefficient is 22. Here is pseudocode for polynomial addition:
You may use ordinary arithmetic operations on individual terms of polynomial equations without defining them yourself. For the above example, the algorithm uses addition on the terms tp and tr. Furthermore, after defining an algorithm, you may use it to define other algorithms. For example, if helpful, you may use polynomial addition in your algorithms for multiplication and division. Be sure your types are correct: if addition is defined over terms, and is defined over polynomials, that does not mean you can add a term to a polynomial unless you have also defined that case.
Answer the following questions:
Be sure to show the values of all variables in your pseudocode at the beginning of each loop iteration.
Here is an example illustration of the addition algorithm running on (2x^2 + 5) + (3x^2 - 4x):
(Notice that the values of p and q did not change throughout the execution of the algorithm. Thus, this algorithm works when p and q are required to be immutable (unchanged). You will learn about immutable objects as you progress on this homework.)
Since your answer will be ASCII text, you won't be able to use italics and subscripts in your answer. You can dispense with italics, and you can represent "tp" as "t_p".
Problem 1 does not involve writing code, but you do have to answer written questions.
Read the specifications for RatNum
,
a class representing rational numbers. Then read the provided
implementation, RatNum.java
.
You will likely want to look at the code in
RatNumTest.java
to see example usages of the RatNum class (albeit in the context of a test
driver, rather than application code).
Answer the following questions, writing your answers in the file hw2/answers.txt. Two or three sentences should be enough to answer each question. For full credit, your answers should be short and to the point. Answers that are excessively long or contain irrelevant information will not receive full credit.
// Rep Invariant for every RatNum r: ( r.denom >= 0 )
List the method or constructor implementations that would have to change. For each changed piece of code, describe the changes informally, and compare the advantages and disadvantages (in terms of code clarity and/or execution efficiency) of each change. Note that the new implementations must still adhere to the given spec; in particular, RatNum.toString() needs to output fractions in reduced form.
return new RatNum ( numerExpr ,
denomExpr);
. Imagine an implementation of the same function
except the last statement is:
this.numer = numerExpr; this.denom = denomExpr; return this;
For this question, pretend that the this.numer and this.denom fields are not declared as final so that these assignments compile properly. How would the above changes fail to meet the specifications of the methods and fail to meet the specifications of the RatNum class? (Hint: Look at the @spec.requires and @spec.effects clauses, or lack thereof.)
Read the specifications for the RatTerm class, making sure you understand the overview for RatTerm and the specifications for the given methods.
Read the provided skeletal implementation of RatTerm.java
,
especially the comments describing how you are to use the provided
fields to implement this class.
Fill in an implementation for the methods in the specification of RatTerm. You may define new private helper methods as you like. You may not add public methods; the external interface must remain the same.
Throughout this assignment, if you define new methods, you must specify them completely. You can consider the specifications of existing methods (where you fill in the body) to be adequate. You should document any code you write, but please do not over-document.
We have provided a checkRep() method in RatTerm that tests whether or not a RatTerm instance violates the representation invariant. We recommend you use checkRep() where appropriate in the code you write. Think about the issues discussed in the last question of Problem 1 when deciding where to call checkRep.
We have provided a fairly rigorous test suite
in RatTermTest.java
. You
can run the test suite to evaluate
your progress and the correctness of your code.
Answer the following questions, writing your answers in the file hw2/answers.txt. Again keep your answers to 2-3 sentences.
RatTerm.toString()
still
cannot produce a term with a zero coefficient (except 0).
coeff.isNaN() ==> expt = 0.
This means that the method implementations could assume this invariant held on entry to the method, but they would also be required to enforce the invariant on exit. List which method or constructor implementations would have to change. For each changed piece of code, describe the changes informally, and compare the advantages and disadvantages (in terms of code clarity and/or execution efficiency) of each change. Note that the new implementations must still adhere to the given spec (except for the part where terms like NaN*x^74 are explicitly allowed).
Which set of RatTerm invariants (coeff.isNaN() ⇒ expt = 0; coeff.equals(RatNum.ZERO) ⇒ expt = 0; both; neither) do you prefer? Why?
Following the same procedure given in Problem 2, read over the specifications for the RatPoly class and its methods and fill in the blanks for RatPoly.java. You don't have to answer any questions in file hw2/answers.txt, as you did for Problem 2. The same rules apply here (you may add private helper methods as you like). Since this problem depends on Problem 2, you should not begin it until you have completed Problem 2 and the hw2.RatTermTest test suite runs without any errors.
You may also want to look at the specifications for the java.util.List class, especially the get(), add(), set(), and size() methods.
You are welcome to do what you like with the provided private helper methods in RatPoly (scaleCoeff and the like). You may implement them exactly as given, implement variants with different specifications, or even delete them; and you may add your own private helper methods. However, you must make sure that every private helper method in the final version of the class has an accurate specification and is not still an unimplemented skeleton.
Make sure your code passes all the tests in RatPolyTest.java
.
Following the same procedure given in Problem 3, read over the specifications for
the RatPolyStack class and its methods and fill in
the blanks for
RatPolyStack.java
. The same rules apply here (you may add private
helper methods as you like). Since this problem depends on Problems 2
and 3, you should not begin it until you have completed Problems 2 and
3 and the hw2.RatTermTest and
hw2.RatPolyTest test
suites run without any errors.
Make sure your code passes all the tests in RatPolyStackTest.java
.
Now that RatPoly and RatPolyStack are finished, you can run the calculator application. This allows you to input polynomials and perform arithmetic operations on them, through a point-and-click user interface. The calculator graphs the resulting polynomials as well.
When you run hw2.CalculatorFrame, a window will pop up with a stack on the left, a graph display on the right, a text area to input polynomials into, and a set of buttons along the bottom. Click the buttons to input polynomials and to perform manipulations of the polynomials on the stack. The graph display will update on the fly, graphing the top four elements of the stack.
Submit your four favorite polynomial equations, in the RatPoly.toString format, in the file hw2/answers.txt.
By the end of the homework, you should have the following files committed and pushed to GitLab in your hw2 directory:
to turn in your assignment, see the Assignment Submission handout.
If you're having trouble running JUnit tests in IntelliJ, try running them from the command line. Instructions are in the Editing and Compiling handout.
All the unfinished methods in RatTerm, RatPoly and RatPolyStack throw
RuntimeExceptions.
When you implement a method, be sure to
remove the throw new RuntimeException();
statement.
To help you find them,
we have also added a TODO: comment in each of these
methods.
Think before you code! The polynomial arithmetic functions are not difficult, but if you begin implementation without a specific plan, it is easy to get yourself into a terrible mess.
The most important method in your RatPoly class will probably be sortedInsert. Take special care with this method.
The provided test suites in this homework are the same ones we will use to grade your implementation. You can consider the provided set of tests to be rigorous enough that you do not need to write your own tests. (In future homeworks you will need to write tests.)
Division of rational polynomials is similar to long division as taught in grade school. We draw an example here:
In this example, the result of truncating division is 1⁄3 x2 - 2⁄9 because truncating division discards the remainder.