🔖 Background Information
Complex numbers have the form where is called the real part of the number, is called the imaginary part of the number, and is the square root of -1. Another way of thinking of is . Some mathematical operations are defined on complex numbers as follows:
Addition:
Subtraction:
Multiplication:
Division:
Negation:
Magnitude:
Conjugation:
Real Part:
Imaginary Part:
This problem is available courtesy of Professor Jason James (James, 2017).
🎯 Problem Statement
Write a ComplexNumber
class that implements operations on complex numbers. The class will include some extra methods that improve quality of life for developers (defined in the Acceptance Criteria).
✅ Acceptance Criteria
- You cannot use any existing complex number type or library - you must implement your own class from scratch.
- Write a class called
ComplexNumber
that takes in two floats as arguments (the real part and the imaginary part of the number). - Implement addition, subtraction, negation, magnitude, conjugation, real, and imaginary methods on the
ComplexNumber
class as instance methods.- Each of these methods should take in no arguments or one argument of type
ComplexNumber
. - Each of these methods should have a return value of type
ComplexNumber
orfloat
, as needed. - None of these methods should change the calling instance of
ComplexNumber
or any arguments.
- Each of these methods should take in no arguments or one argument of type
- Implement a
prettyPrint
method on theComplexNumber
class that prints the complex number out to the console in the form “a + bi”.
📋 Dev Notes
- You need to think carefully about how you want to test this class. How can you implement unit tests on your custom class? Maybe, you could consider testing the real and imaginary part separately?
- You do NOT have to put your
ComplexNumber
class into another file using headers. However, there is an add-on for the portfolio that you can do for extra credit.
🖥️ Example Output
Within your C++ code, you might have some examples that look like this:
ComplexNumber numberOne = ComplexNumber(5.0, 7.0);
ComplexNumber numberTwo = ComplexNumber(2.5, 3.1);
ComplexNumber numberThree = numberOne.add(numberTwo);
cout << numberThree.prettyPrint(); // Returns 7.5 + 10.1i
ComplexNumber numberFour = numberOne.subtract(numberTwo);
cout << numberFour.prettyPrint(); // Returns 2.5 + 3.9i
float magnitude = numberOne.magnitude();
cout << magnitude; // Returns 8.602325267
📝 Thought Provoking Questions
- Does it make sense to default any of the arguments in the constructor of
ComplexNumber
to a particular value? Why or why not? - What are the types of values that are returned from each method in
ComplexNumber
? - What happens if you try to initialize your
ComplexNumber
class with an integer instead of a float? - Does
prettyPrint()
print anything besides the number in “a + bi” form (even anendl
)? Why or why not?
💼 Add-Ons For the Portfolio
(One Credit) Calculate the Phase
Add a method to your ComplexNumber
class called phase
which calculates the phase of the number. The phase of the number is given by:
Be sure that the method is tested either by your driver function or assertions.
(Two Credits) Header File
Move your ComplexNumber
class into a separate file from main()
and create a corresponding header file. Then, #include
the header file for your code in main()
. This should not change the functionality of your code.
(Two Credits) Multiplication and Division
In the “Background Information” section, we defined multiplication and division of complex numbers. However, we did not implement them in the lab, originally.
Implement two methods on the ComplexNumber
class for multiplication and division. Just like with the other operations:
- Each of these methods should take in one argument of type
ComplexNumber
. - Each of these methods should have a return value of type
ComplexNumber
. - Neither of these methods should change the calling instance of
ComplexNumber
or any arguments.
🔗 Useful Links
N/A