Subgoals - Evaluating Expressions
One of the unique parts of this course will be the inclusion of subgoals to help us better understand what is going on in our code. Subgoals are designed to help us structure our thinking process in a way that matches what an experienced developer might have.
A great example is learning how to read. Fluent readers can read whole words at a time, while processing and retaining information from several sentences at once. Novice readers have to sound out words one letter at a time, and can store and retain a few words at a time in memory before being overwhelmed by information. Learning a new language can quickly make this difference very apparent, even for readers who are fluent in many languages.
Writing computer code is very similar. An experienced developer can fluently read and write code, seeing and understanding entire code structures at a glance. Novice developers often have to look at a single line at a time, and break it down into individual parts before it is apparent what the code says. Thatās just part of the learning process!
To make this process easier, this course will include many pages that describe subgoals we can use to help analyze and write code. In essence, we can treat these subgoals as a set of steps to follow mentally when performing these tasks. Over time, weāll become more comfortable with each task, and the subgoals will quickly become ingrained knowledge. Just like we can easily read both ācatā and ācatastropheā at a glance, with practice weāll be able to read code just as quickly!
For more information on Subgoals, see the work of Dr. Briana Morrison published through the ACM and elsewhere. Link
Analyzing Expressions
To analyze an expression involving variables and mathematical operators in code, here are some mental subgoals we can use:
1. Check Types
First, we must determine whether the data type of the expression is compatible with the data type of the variable we want to store it in. In Java, we must be very careful to make sure we are only storing whole numbers in the integer data type, and floating point values in the double data type.
2. Perform Prefix Operations
Next, we should look at the expression to determine if there are any prefixed operations that must occur first. In Java, for example, we could find a prefixed increment operator like ++x
, so weāll need to update the value of x
before moving to the next step.
3. Solve Arithmetic Equation
At this point, we can solve the arithmetic equation using the order of operations for our language. This simply involves the process of substituting values for variables and performing the requested operations. However, once again we must be careful to make sure that the operands provided to each operator are valid and produce the correct data type. The example we saw earlier for handling a large equation showed us a great way to work through this process.
4. Confirm Data Type of Result & Store It
Once weāve solved the arithmetic equation, we should be left with a variable on the left side of the equals sign and a single value on the right side. So, once again, we should confirm that the value on the right can be stored in the data type of the variable on the left.
5. Perform Postfix Operations
Finally, if the original expression included any postfix operations, such as a postfixed decrement like x--
in Java, weāll need to update the value of x
before moving on to the next line.
Examples
Letās walk through an example showing how to use these subgoals to evaluate a few expression statements.
int x = 3 + 5;
Weāll break this down by subgoals:
- We can easily confirm that the data type of the variable is
int
, and that the numbers on the right side of the equation are all integers, so we shouldnāt have any issues with data types initially. - This reminds us to look for prefix operations. Since there arenāt any, we can move on.
- We can now solve the arithmetic equation on the right side of the expression. So, we can easily find that
3 + 5
can be replaced with8
. - This tells us to once again confirm that the result is a compatible value for the data type of the variable on the left. Since
8
can be stored in anint
variable, we are just fine. So, we can update the value ofx
to8
. - Finally, we must look for any postfix operations in the original equation. Since there arenāt any, we can skip this step.
Thatās all there is to it! While this may seem like quite a bit of information for handling a simple statement, it becomes much more useful as the statements get more complex.
Hereās a second example:
int x = 5;
double y = 3.5;
double z = ++x + -y--;
Once again, letās break the last line in this code example down by subgoals to evaluate the expression:
- This is a bit tricky, since the statement includes several variables. However, we can look at the earlier code to see that
x
is an integer type, andy
is a double type. Since this expression is adding them together, the result should be a floating point number, which can be stored back in the variablez
with type double. So, we should be fine here. - Next, weāll see that there is a prefix expression
++x
, so weāll increment the value ofx
by to . We also see a prefix operator-y
, so weāll need to remember that we are dealing with the inverse of the value stored iny
. - Now we can solve the equations. To do this, weāll replace each variable with its value, then calculate the result. So,
x + -y
becomes6 + -3.5
which results in2.5
. - Once we have the result, we must confirm that it can be stored in the variable to the left. Since
2.5
is a floating point number, it can be stored in the double data type. So, weāll update the value ofz
to . - Finally, weāll need to look for any postfix operators in the original equation. Weāll see
y--
, so we must decrement the value ofy
by to
So, at the end of this short code example, weāll see that x
is
y
is
z
is also
Finally, letās look at one more example:
int x = 5;
double y = 8.5;
int z = x + y;
Hereās how we can break down the process of evaluating the last line in this example:
- First, just like the example above, we see that
x
is an integer andy
is a double, so the sum of those values should also be a double. However, weāll see thatz
is an integer. So, what will happen here? As it turns out, the Java compiler will give us an error when we try to compile this code, because we are assigning a double value to an integer which could result in the loss of data.
As we can see, by carefully following these subgoals, we can even find errors in our code, just by evaluating it step by step.