2019-04-14
Adding numbers using Boolean operations in JavaScript
javascript, microprocessors
javascript, microprocessors
Photo by Crissy Jarvis_ on _Unsplash
You know how to add numbers progmatically, right?
1 + 1 will basically give you 2.
Numbers are added in binary form down in machine level.
But how do numbers get added underneath the hood?
I will show how to add "positive" integers (no floating) using boolean operations.
I will assume the knowledge of binary numbers and Boolean operations.
And you can follow along on CodeSandbox.
Below is the truth table of all possible XOR & AND operations I will refer back to.
AND & XOR
When you add two one-bit numbers, you get either 0 or 1 for the sum and 0 or 1 for the carry.
Adding one bit numbers
Did you notice that, carry output looks the same as output of AND truth table, and sum equal to that of XOR?
The operation can be represented using logical XOR & AND gates as shown here.
Half-adder
A circuit formed that way is called half-adder.
Armed with the knowledge and we can now implement the addition using XOR & AND.
https://gist.github.com/dance2die/6a002c4d417703cc31c0c02f8bbfe067
When we run the half adder on combination of one bit addition, the result looks like below.
https://gist.github.com/dance2die/562a0e63f8451de351ec3d3a4eb5c494
OK, that wasn't interesting enough as we can't do anything by adding just one bit.
Let's spice it up by adding two bits.
We got the carry from the half-adder but to calculate the next bit we need to pass the carry to the next adder.
But the problem is that, half-adder accepts only two inputs and doesn't accept a carry.
We can solve the problem by combining two half-adders, making it a full-adder.
Logic looks like following.
Simply put, you perform two operations. One for the current bit, and another with the carry.
Let's take a look at an example of adding 11 and 01 to get 100.
Result of 11 + 01 => 100
I apologize for the 💩 illustration 😅.
And thank you @MarkN_LP for catching the error.
The diagram shows the result of first carry being fed into the 2nd half-adder, which is used to calculate sum.
Let's implement the full-adder & add two bit numbers.
https://gist.github.com/dance2die/cee0a284dcf0be414a2974f567140e7d
Full-adder is implemented in line#4~8 using newly created orBit method to calculate the carry.
It uses two half-adders and uses the carry from the "first" operation in the second half-adder.
And the carry is the result of two carries in the two half-adders as shown in the diagram.
11 + 01 correctly returns { c1: 1, b1: 0, b0: 0 }.
Still useless right? Let's add more bits.
When you add one bit, you need just a half-adder. For two bits, 1 half-adder & 1 full-adder.
For 3 bits, you would need 1 half-adder & 2 full-adders.
So for N-bit addition, you need 1 half-adder & N-1 full-adders.
I could've shown 3-bit operation but decided to share a method that works on any N-bits instead (unlike how microprocessors are physically constrained).
https://gist.github.com/dance2die/3f11e90723410431aae22f18fb4b0124
This code assumes that length of two digits have the same length.
I initially was going to change the length dynamically but it made the demo code too convoluted so left it out.
Line #2 & #3 converts strings into array of numbers
and #7 uses reduceRight to start working on the least significant (right-most) bit.
On first iteration, we calculate the sum using half-adder on line #14, and then we use the full-adder for the rest.
Carry passed to the full-adder is retrieved from the first item in the array because we are prepending new digit ([{c, s}, ...acc]) on each iteration.
Lastly, we are returning a text representation of the sum for demo purpose only.
Sorry about abusing _&&_ there 😜.
I got excited after reading "Mastering JavaScript’s && and || logical operators" by Nicolas Marcora today. 🙂
Let's check out the demo result.
https://gist.github.com/dance2die/0d9666493c1d634547d070419103721b
Values within parenthesis shows operations in base 10.
We've looked at how positive numbers are added under the hood.
I am also just learning about this thus the explanation might lack much.
The source I am learning from is "The Manga Guide to Microprocessors".
I still haven't finished the book but it has been delightful.
If you want to dig deeper, check out following resources.