CS II Lesson 11

Professor Abdul-Quader

(Im)mutability

Design discussion

Problem: Ask the user to input in two GaussianIntegers (their real parts and their imaginary parts). Create the two GaussianIntegers, output them, add them, and then output their sum.

Recall: A Gaussian integer is a complex number of the form \(a + bi\), where \(a\) and \(b\) are integers (\(i\) is the imaginary number representing a square root of \(-1\)). How would you design a Gaussian integer?

OOP

Question: What are the nouns of this problem?

  • Real parts and imaginary parts of each GaussianInteger?
  • Just the GaussianIntegers?

Object-orientation

How should we design the add method? Option 1:

public void add(int otherReal, int otherImag)

Option 2:

public void add(GaussianInteger other)

Which is more “object-oriented”?

Implementation

What data is required to represent a Gaussian Integer? What are the actions it should be able to do?

Exercise (Part 1)

  • Create a GaussianInteger class.
  • Declare your instance variables.
  • Create a constructor for that class
  • Create the toString method.
  • In a separate class, make a main method that instantiates a GaussianInteger object and prints it out.

Exercise (Part 2)

Create a method to add another GaussianInteger object. To add two GaussianIntegers, add their real parts and add their imaginary parts (where do you store them?). Test it out in your main method:

GaussianInteger g1 = new GaussianInteger(2, 3); // 2 + 3i
GaussianInteger g2 = new GaussianInteger(3, -5); // 3 + 5i
System.out.println(g1);
System.out.println(g2);
g1.add(g2); // Now g1 should be 5 + -2i
System.out.println(g1);

Exercise (Part 3)

Now create a method to multiply another GaussianInteger object. Test it out in the main method.

Hint: If \(g1 = a + bi\), and \(g2 = c + di\), then \(g1 \times g2 = (a + bi)(c + di) = (ac - bd) + (ad + bc)i\)

// test it out in main:
GaussianInteger g = new GaussianInteger(1, 2); // 1 + 2i
GaussianInteger x = new GaussianInteger(3, 4); // 3 + 4i
g.multiply(x); // g should now be -5 + 10i
System.out.println(g);

Aside: Debugging

Why doesn’t the below multiply work:

public void multiply(GaussianInteger other) {
  // real part: ac - bd
  real = real * other.real - imag * other.imag;
  // imaginary part: ad + bc
  imag = real * other.imag + imag * other.real;
}

Let’s try out the debugger to see what happens.

Aside: Debugging

The debugger in IntelliJ is actually a bit easier to use:

Immutability

Recall: an object is mutable if its data (instance variables) can be changed after the object was created. It is immutable otherwise. (Strings are immutable; ArrayLists are mutable.)

Our GaussianInteger

Was our implementation of GaussianInteger mutable or immutable? Consider the following code:

GaussianInteger gi = new GaussianInteger(3, 4);
int something = SomeLibrary.libraryMethod(gi);
System.out.println(gi); // what will this output?
  • Hope: it should output “3 + 4 i”. Will that always happen?
  • Compare to what happens with integer parameters.

Benefits of mutable objects:

  • Flexibility.
  • Easier to design / reason about
  • Sometimes more efficient?

Benefits of immutable objects:

  • Safety! The object can be shared without worrying about what other methods / threads do to it.
  • Easier to reason about.
  • Other benefits require more material that we haven’t covered (yet): handles errors better, easier to parallelize, better to use with Map and Set types, \(\ldots\)

Immutable GaussianInteger

To re-design GaussianInteger to be an immutable type, first we can try to ensure that the instance variables can’t be changed: e.g. make them all final! First try that.

Any errors? They tell us what we need to change:

  • add and multiply.
  • Instead of making them void, have them return new GaussianInteger objects.
  • Now we don’t change our original object!

Changing add method

public GaussianInteger add(GaussianInteger other) {
    GaussianInteger gi = new GaussianInteger(...); // fill this in
    return gi;
}

Exercise

Re-design your GaussianInteger class to be immutable. Test it out with the following main method:

GaussianInteger one = new GaussianInteger(1, 0);
GaussianInteger i = new GaussianInteger(0, 1);

System.out.println(one); // should be 1 + 0i
System.out.println(i); // should be 0 + 1i
System.out.println(one.add(i)); // should be 1 + 1i
System.out.println(one); // should still be 1 + 0i
System.out.println(i.multiply(i)); // should be -1 + 0i
System.out.println(i); // should be 0 + 1i

Kinds of errors

  • Compiler errors: syntax errors, type mismatches, etc. These are caught by the compiler.
  • Run-time errors: usually Exceptions, but also things like OutOfMemoryError.
  • “Bugs” or logic errors: when the program doesn’t do what you wanted it to do. Focus today: these types.

Examples for today are taken from Bloch and Gafter, Java Puzzlers: Traps, Pitfalls and Corner Cases (2005).

Set-up

Exercises for today are here.

Example: Increment

int j = 0;
for (int i = 0; i < 100; i++) {
    j = j++;
}
System.out.println(j);
  • Before running, what do you think will happen?
  • Run the program. What happens?

Exercise

  • Figure out what happened.
  • Put print statements in the for loop to see what the value of \(j\) is

Debugger

Modern IDEs have built-in tools to help us debug.

  • Click on the “side” of line 6 to add a breakpoint.
  • Use the debugger.
  • The program will run until it hits the breakpoint, and then stop and let us continue execution one line at a time.
  • The debugger shows us a memory diagram, the values of all the variables at a particular moment. What happens to \(j\) as \(i\) increases?

Wrap-up Example 1

  • Assignment statements also can be used as expressions!
  • The value of j++ is the value of \(j\) before the increment.
  • The line j = j++; returns the previous value of \(j\) and assigns that back to itself.

Lesson: Never assign to the same variable more than once in a single expression.

More Examples

There are two other examples in the LogicErrors code:

  • InTheLoop
  • AnimalFarm

Go through these on your own; we will discuss next time.

InTheLoop

Look at the InTheLoop program:

public static final int END = Integer.MAX_VALUE;
public static final int START = END - 5;

public static void main(String[] args) {
    int count = 0;
    for (int i = START; i <= END; i++)
        count++;
    System.out.println(count);
}
  • Before running, what do you think will happen?
  • Run the program. What happens?
  • Use the debugger to figure out what happened.

AnimalFarm

final String pig = "length: 10";
final String dog = "length: " + pig.length();
System.out.println(pig);
System.out.println(dog);
System.out.println("Animals are equal: " + pig == dog);
  • Before running, what do you think will happen?
  • Run the program. What happens?
  • Use the debugger to figure out what happened.

Upcoming

  • Thursday:
    • Finish demos
    • More on debugging
    • Quiz 3
  • Project 2 due 3/31
  • Exam 2 will be after Spring Break (4/3).
// reveal.js plugins