Luckily we donβt have to write all the methods we might need ourselves. Java comes with a rich library of methods that we can use. (In fact weβve been using a few of these methods already, print and println. Weβll explain in a later lesson what the System.out part is about.)
In a future lesson weβll talk about how to explore the library of methods that come with Java. For now we want to look at one particular set of methods that are useful for doing math beyond the basic arithmetic we can do with the arithmetic operators weβve already learned about.
As we learned in the ChapterΒ 1Β Introduction, all code in Java is written within classes. Thereβs a lot to say about classes and weβll get to it in later units. But some classes just exist as a place to put a bunch of useful methods. One such class is the Math class in the java.lang package which is part of every Java program. (Weβll also learn more about packages in later units.)
The Math class contains five methods that are in the AP CSA subset and are listed in the AP CSA Java Quick Reference Sheet that you will have access to during the exam.
Math.abs is an example of an overloaded method meaning there are two methods with the same name but different signatures: one that takes an int and one that takes a double. They each return the absolute value, i.e. the positive value of the number without its sign, of their argument. But the return type of each method also differs, being the same same as the type of the parameter.
Math.pow takes two arguments, both doubles and returns a double which is the first argument raised to the power of the second argument. I.e. pow(a, b) is what we would write in math as \(a^b\text{.}\) But because ints can be silently converted to doubles, we can call Math.pow with int arguments and they will automatically be converted to doubles. As a result, the return value will always be a double even if it could be represented by an int.
Math.pow(2 , 3); // returns 8.0, not 8
Math.pow(10, 6); // returns 1000000.0, not 1000000
Math.pow(2, -3); // returns 0.125
Note3.3.1.
If you are a Desmos or graphical calculator user, note that ^ is not an exponentiation operator in Java. Worse luck, it is an operator on ints but it does something completely different. So if a and b are ints and you write a ^ b, intending to raise a to the bth power, the compiler wonβt complain but your code will do something very different. If you want an int result, use a cast: (int) Math.pow(a, b). Though watch out for overflow errors since even relatively small exponents can produce values greater than Integer.MAX_VALUE.
Math.sqrt takes an double argument and returns a double value which is the nearest approximation of the positive square root of the argument. As with Math.pow, we can call it with an int argument but the argument will be converted to a double and the result will always be a double even if it could have been represented with an int.
Math.sqrt(2); // returns 1.4142135623730951
Math.sqrt(9); // returns 3.0, not 3
Since these methods are all non-void methods, calling them only makes sense in a context that will do something with the value they return, such as assigning the value to a variable or printing it out.
The distance between two numbers on the number line is defined as the absolute value of their difference. Their difference is just what you get when you subtract one from the other. For example, the distance from 0 to 3 is 3, the distance from -3 to 0 is 3, and the distance from -3 to 1 is 4.
Using the definition of distance from the previous problem, fill in the method distance below so it correctly computes the distance between two numbers a and b.
Games would be boring if the same thing happened each time you played the game. And many simulation techniques, including those used to train Large Language Models like ChatGPT, depend on generating random variations. The key to making games interesting and to running useful simulations is the ability to generate random numbers. The Math.random() is the simplest way to get random numbers in Java.
Math.random() is not a function in the mathematical senseβit doesnβt return the same value every time it is called with the same arguments. In fact, it doesnβt even take any arguments. But every time we call it we will get a double greater than or equal to 0.0, and less than 1.0. For all practical purposes the number we get each time is random and all numbers that can be represented by a double between zero and one are equally likely.
When we talk about ranges of numbers sometimes we need to be precise about whether the ends of the range are part of the range. For example, loosely speaking we might say that Math.random returns a number between zero and one. But does that mean it can return exactly 0.0? Or exactly 1.0? As it turns out it can return 0.0 but never returns 1.0.
When we need to be precise about this weβd say that it returns a number between zero, inclusive, and one, exclusive, meaning include zero but exclude one. Lots of ranges in Java are expressed this way, as youβll see later on with an inclusive bottom and exclusive top.
Getting a number in the range from zero to one, may not seem all that useful by itself. But we can expand the range easily enough. To see how, imagine you had less than a dollar to your name and you wanted to be richerβyouβd want to find a way to multiply your money. If you could invest every penny you had in something that would multiply your money by 1,000 then instead of having somewhere between $0 and $1, then youβd have somewhere between $0 (inclusiveβif you started with $0) and $1,000 (exclusive, since if you had even a fraction of a penny less than $1 multiplying by 1,000 would still leave you just a bit shy of $1,000.) If the investment multiplied your original money by a million, youβd have between $0 and $1,000,000! (But never quite $1,000,000.)
Same trick applies to random numbers. The value Math.random returns is like the initial amount of money in your pocket, always a bit less than $1. If you multiply that value by any amount, it will stretch it into the range you want:
You may have noticed that while the numbers generated were always in the range 0 to 10, all the numbers probably had a lot a digits after the decimal point. Often we want a random integer, with nothing after the decimal point. Easy enoughβcasting a double to an int will throw away any values after the decimal point. For example,
// rnd will be an integer in the range 0-9 (from 0 up to 10).
int rnd = (int) (Math.random() * 10);
Finally, what if we want a number in a range that doesnβt start with 0, say a number from 1 to 10 (including 10) instead of from 0 to 9 (including 9)? Since the size of the two ranges is the same, with ten numbers in each, all we need to do is shift from the range weβve generated into the range we want. In other words, add the difference between the two ranges, 1 in this case.
// rnd will be an integer in the range 1-10 (including 10).
int rnd = (int) (Math.random() * 10) + 1;
Activity3.3.7.Random ints.
Run the code below several times to see how the value changes each time. How could you change the code to return a random integer from 1 to 10? Modify the code and see if your answer is correct. Try removing the parentheses from around (Math.random() * 10) and run the code several times. What happens? The parentheses are necessary because (int) will cast the closest expression, and (int)Math.random() will always be 0 since anything after the decimal point is dropped.
// Math.random() returns a random number between 0.0 (inclusive) and 1.0 (exclusive)
double rnd = Math.random();
// rnd1 is an integer in the range 0-9 (including 9).
int rnd1 = (int) (Math.random() * 10);
// rnd2 is in the range 1-10 (including 10).
int rnd2 = (int) (Math.random() * 10) + 1;
// rnd3 is in the range 5-10 (including 10). The range is 10-5+1 = 6.
int rnd3 = (int) (Math.random() * 6) + 5;
// rnd4 is in the range -10 up to 10 (including 10). The range is doubled (10 - -10 + 1 = 20) and the minimum is -10.
int rnd4 = (int) (Math.random() * 21) - 10;
So the general recipe for generating a random is to first stretch the value from Math.random() until itβs in a range of the right size by multiplying by the size of the range. Then if we want an integer value, cast to int to discard the part after the decimal point. Then shift the value up by adding the minimum value. The table below shows some applications of that general recipe.
Remember that (int) (Math.random() * size) + min takes random numbers in the range 0-size to a new range starting at min. We want the minimum number to be 25, but the minimum number here would be 36.
Remember that (int) (Math.random() * size) + min takes random numbers in the range 0-size to a new range starting at min. We want the minimum number to be 25, but the minimum number here would be 60.
Remember that (int) (Math.random() * size) + min takes random numbers in the range 0-size to a new range starting at min. We want the minimum number to be 25, but the minimum number here would be 60.
Yes, (int) (Math.random() * 36) + 25 takes random numbers in the range from 0-36, exclusive, into the range 25-60 (inclusive). The range is (max number - min number + 1) which is (60 - 25 +1) = 36.
You may have a combination lock on your locker at school where you have to spin the dial to three separate numbers from 0 up to 39, inclusive. What if you forgot your combination? Would you be able to guess it?
Write code that will generate 3 random integers from 0 up to 40 (but not including 40) using Math.random() in the Active Code window below. Run it a couple times to see it generate different numbers.
How many times would you need to run it to guess your combination correctly? Letβs have the code compute the number of permutations possible in your combination lock using Math.pow(number, exponent). For example, if you had to spin the dial twice on your combination lock where each spin can choose a digit from 0-9 (10 digits), there are \(10^{2}\) possible permutations. In Java, this would be written as Math.pow(10, 2) which means 10 to the power of 2. If you start listing all the permutations possible, you can tell that there are \(10^{2}\) or 100 possible permutations for a 2 spins on a dial lock from 0-9.
Now what about the combination lock for this challenge? You will need to spin the dial 3 times: once to the right, once to the left, and once to the right to three different numbers from 0 up to 40 (not including 40). In general, the formula to use is \(n^{s}\) where \(n\) is the number of numbers on the dial and \(s\) is the number of spins. Write this using the Math.pow() method in your code and save it into a variable and print out.
Hereβs another challenge that is a lot of fun! Can you use random numbers to make dancing turtles? This idea was suggested by CSA teacher Zac Martin.
Complete the random numbers using Math.random() in the correct ranges to choose x, y coordinates and a random color by choosing random red, green, and blue values in the range of 0-255. Put on some music and watch your turtle dance!
The Pythagorean theorem states that the length of the hypotenuse (the side opposite the right angle in a right triangle) is the square root of the sum of the squares of the lengths of the other two sides, also called the βlegsβ or the width and height of the triangle. Hereβs the formula for this theorem: \(c = \sqrt{a^{2} + b^{2}}\) where \(a\) and \(b\) are the width and height of the triangle and \(c\) is the length of the hypotenuse.
This may seems like a bunch of Greek to you but one common, real-world use for the Pythagorean theorem is to calculate the length of ladder you will need to reach the window of your beloved, given that their cruel parents have locked them in a tower surrounded by a moat. The ladder will be the hypotenuse of a triangle whose legs are the height of the window of your belovedβs room in the tower and the width of the moat since you have to place the base of the ladder on the edge of the moat.
Which of the following are correct Java expressions to compute the hypotenuse of a triangle with legs a and b given the Pythagorean Theorem \(c = \sqrt{a^{2} + b^{2}}\) where \(a\) and \(b\) are the lengths of the legs and \(c\) is the length of the hypotenuse?
β This is the sum of the squares of the lengths of the legs which gives us the square of the hypotenuse. We need a Math.sqrt to get the length of the hypotenuse.
Complete the ladderSizeNeeded method below using the Pythagorean Theorem and the Math.sqrt method. Then in the main method, write a method call to test the ladderSizeNeeded method with arguments for the height of 30 and the width of 40. The method should return the length of the ladder needed to reach the window of your beloved.
(AP 1.10.A.2) Class methods are typically called using the class name along with the dot operator. When the method call occurs in the defining class, the use of the class name is optional in the call.
(int)(Math.random() * range) + min moves the random number into a range starting from a minimum number. The range is the (max - min + 1). For example, to get a number in the range of 5 to 10, use the range 10-5+1 = 6 and the min number 5: (int)(Math.random()*6) + 5.