Recently, I have been revisiting Solidity, consolidating the finer details, and writing "WTF Solidity" tutorials for newbies.
Twitter: @0xAA_Science | @WTFAcademy_
Community: Discord|Wechat|Website wtf.academy
Codes and tutorials are open source on GitHub: github.com/AmazingAng/WTFSolidity
In this section, we will introduce control flow in Solidity, and how to use Solidity to implement insertion sort (InsertionSort
), a program that looks simple but is actually bug-prone.
Solidity's control flow is similar to other languages, mainly including the following components:
if-else
function ifElseTest(uint256 _number) public pure returns(bool){
if(_number == 0){
return(true);
}else{
return(false);
}
}
for loop
function forLoopTest() public pure returns(uint256){
uint sum = 0;
for(uint i = 0; i < 10; i++){
sum += i;
}
return(sum);
}
while loop
function whileTest() public pure returns(uint256){
uint sum = 0;
uint i = 0;
while(i < 10){
sum += i;
i++;
}
return(sum);
}
do-while loop
function doWhileTest() public pure returns(uint256){
uint sum = 0;
uint i = 0;
do{
sum += i;
i++;
}while(i < 10);
return(sum);
}
- Conditional (
ternary
) operator
The ternary
operator is the only operator in Solidity that accepts three operands:a condition followed by a question mark (?
), then an expression x
to execute if the condition is true followed by a colon (:
), and finally the expression y
to execute if the condition is false: condition ? x : y
.
This operator is frequently used as an alternative to an if-else
statement.
// ternary/conditional operator
function ternaryTest(uint256 x, uint256 y) public pure returns(uint256){
// return the max of x and y
return x >= y ? x: y;
}
In addition, there are continue
(immediately enter the next loop) and break
(break out of the current loop) keywords that can be used.
Note: Over 90% of people who write the insertion algorithm with Solidity will get it wrong at first try.
The problem solved by the sorting algorithm is to arrange an unordered set of numbers from small to large, such as sorting [2, 5, 3, 1]
to [1, 2, 3, 5]
. Insertion Sort (InsertionSort
) is the simplest sorting algorithm and the first algorithm many people learn in their computer science class. The idea of InsertionSort
is very simple: from begin to the end, compare each number with the number in front of it. If it is smaller than the number in front, switch their positions.
The schematic is shown below:
data:image/s3,"s3://crabby-images/56c2d/56c2dfaa5c9622cca940b196fb9ecc193772058b" alt="InsertionSort"
We will first look at the Python implimentation of Insertion Sort:
# Python program for implementation of Insertion Sort
def insertionSort(arr):
for i in range(1, len(arr)):
key = arr[i]
j = i-1
while j >=0 and key < arr[j] :
arr[j+1] = arr[j]
j -= 1
arr[j+1] = key
Python version of Insertion Sort takes up 8 lines. We can rewrite it into Solidity by converting functions, variables, loops accordingly. It only takes up 9 lines of code:
// Insertion Sort (Wrong version)
function insertionSortWrong(uint[] memory a) public pure returns(uint[] memory) {
for (uint i = 1;i < a.length;i++){
uint temp = a[i];
uint j=i-1;
while( (j >= 0) && (temp < a[j])){
a[j+1] = a[j];
j--;
}
a[j+1] = temp;
}
return(a);
}
But when we compile the modified version and try to sort [2, 5, 3, 1]
. BOOM! There are bugs! After long-time debugging, I still could not find where the bug is. I googled "Solidity insertion sort", and found that the insertion algorithm written with Solidity are all wrong, such as: Sorting in Solidity without Comparison
Errors occured in Remix decoded output
:
After a few hours, with the help of a friend in the Dapp-Learning
community, we finally found the problem. The most commonly used variable type in Solidity is uint
, which represent a positive integer. If it takes a negative value, an error underflow
will be reported. In the above code, the variable j
may get -1
, resulting in the corresponding error.
So, we just need to add 1 to j
so that it can never take a negative value. The correct version of Insertion Sort:
// Insertion Sort(Correct Version)
function insertionSort(uint[] memory a) public pure returns(uint[] memory) {
// note that uint can not take negative value
for (uint i = 1;i < a.length;i++){
uint temp = a[i];
uint j=i;
while( (j >= 1) && (temp < a[j-1])){
a[j] = a[j-1];
j--;
}
a[j] = temp;
}
return(a);
}
Result of the correct code:
In this lecture, we introduced control flow in Insertion Sort algorithm in Solidity. Solidity looks simple but have many traps. Every month, projects lose millions of dollars because of small bugs in their smart contract. To write a safe contract, we need to master the basics of the Solidity and keep practicing.