Starklings-Cairo1: Variables in Cairo
Overview
In Cairo, variables are immutable by default.
When a variable is immutable, once a value is bound to a name, you can’t change that value.
You can make them mutable by adding mut
in front of the variable name.
It is however important to clarify the fact that even though the variable can be made mutable, Cairo works with an immutable memory model, meaning that changing the value of a variable will not change the value in memory but rather assign a new memory location to that variable.
Now Available in Video Format
If you prefer watching over reading, a video version of this post is available.
Further information
- Memory model in Cairo - Explains how Cairo handles memory operations, emphasizing its immutable memory model.
- Variables in Cairo - Detailed guide on how variables work in Cairo and how mutability can be applied.
- Integer Types - Overview of integer data types available in Cairo.
Intro
We'll start from the very basics, since this is where the exercises start -- let's get started!
[!Note] For simplicity in presentation, we're denoting code as
rust
in the blog sincecairo
is not yet supported as a syntax highlighting option.
variables1.cairo
// variables1.cairo
// Make me compile!
// Execute `starklings hint variables1` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
fn main() {
x = 5 ;
println!(" x is {}", x)
}
our first exercise shows a basic function that is giving us an error:
variables1.cairo
Errors
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Identifier not found.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:10:5
x = 5 ;
^
error: Invalid left-hand side of assignment.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:10:5
x = 5 ;
^
error: Identifier not found.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:11:26
println!(" x is {}", x)
^
error: Type annotations needed. Failed to infer ?4
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo[println_macro][writeln_macro]:4:22
match core::fmt::Display::fmt(__write_macro_arg0__, ref __formatter_for_print_macros__) {
^*****^
could not compile `exercise_crate` due to previous error
⚠️ Failed to run exercises/variables/variables1.cairo! Please try again.
The errors indicate several issues:
- Identifier not found: The variable
x
is not declared before it is used. - Invalid left-hand side of assignment: The
x
needs to be declared properly. - Type annotations needed: The type of
x
is not inferred correctly by the compiler.
If you're familiar with Rust, you'll quickly spot that the issue is that the let
keyword is missing.
variables1.cairo
Solution
fn main() {
let x = 5 ; // adding `let` before `x`
println!(" x is {}", x)
}
Adding let
as shown above quickly and easily fixes our errors and get's our code to compile.
Explanation
- Declaration with
let
: In Rust and Cairo, variables must be declared before they are used. Thelet
keyword is used for this purpose. It tells the compiler that a new variable is being declared.
let x = 5;
- Type Inference: While Rust and Cairo can infer types in many cases, constants and certain other values require explicit type annotations. In this example, type inference works fine because
5
is clearly an integer.
. Printing Variables: The println!
macro is used to print values. In this case, it prints the value of x
.
println!(" x is {}", x);
variables2.cairo
// variables2.cairo
// Execute `starklings hint variables2` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
fn main() {
let x;
if x == 10 {
println!("x is ten! ");
} else {
println!("x is not ten! ");
}
}
Here we have another simple function with an x
variable as well as an if
statement that compares x
to 10
but it's not compiling. Let's look at our errors:
variables2.cairo
Errors
🟡 Running exercises/variables/variables2.cairo exercise...
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Missing token TerminalEq.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:8:10
let x;
^
error: Missing tokens. Expected an expression.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:8:10
let x;
^
error: Unsupported feature.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:8:10
let x;
^
It's pretty clear that our first line in our function is not correct syntax and should be written differently, the compiler tells us that we're Missing tokens. Expected an expression.
So let's fix it.
variables2.cairo
Solution
As we saw in the first exercise that was missing the let
keyword before x
this exercise is missing the value we want x
to be. We fix that by adding the assignment operator = and an actual value.
let x;
// to
let x = 10;
Now depending on what value we use we'll get different outputs as we can see with the else
part of the if
statement. Feel free to experiment and recompile with different numbers as the x
value and see what happens. But here's a solution with the value 1
. This would of course print: x is not ten!
.
fn main() {
let x = 1;
if x == 10 {
println!("x is ten! ");
} else {
println!("x is not ten! ");
}
}
Explanation
- Variable Declaration and Initialization:
- In the initial code,
let x;
declares the variablex
but does not initialize it with a value, which is why the compiler throws an error. - Correcting it to
let x = 10;
initializesx
with the value10
.
- In the initial code,
- Conditional Statement:
- The
if
statement checks whetherx
is equal to10
. - If
x
equals10
, it prints"x is ten!"
. - Otherwise, it prints
"x is not ten!"
.
- The
- Experimentation:
- Changing the value of
x
to different numbers and recompiling will result in different outputs based on theif
condition.
- Changing the value of
So far so good right, it's pretty straight forward. Let's move on to exercise 3.
variables3.cairo
// variables3.cairo
// Execute `starklings hint variables3` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
fn main() {
let x: felt252;
println!("x is {}", x);
}
Here we see another similar scenario as exercise 2, but we see a different type of syntax the : felt252
after the variable x
. Let's look at the errors.
variables3.cairo
Errors
🟡 Running exercises/variables/variables3.cairo exercise...
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Missing token TerminalEq.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:9:19
let x: felt252;
^
error: Missing tokens. Expected an expression.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:9:19
let x: felt252;
^
error: Unsupported feature.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:9:19
let x: felt252;
^
could not compile `exercise_crate` due to previous error
⚠️ Failed to run exercises/variables/variables3.cairo! Please try again.
The errors indicate that the expression is incomplete. Specifically, the variable x
is declared with a type felt252
but is not initialized with a value.
variables3.cairo
Solution
In Cairo, as in Rust, you can specify types after variables. This is beneficial for clarity and type safety. Although the Cairo compiler often infers the type (defaulting to felt252
if unspecified), it's good practice to explicitly declare types, especially for beginners.
To fix the error, we need to initialize x
with a value by adding the assignment operator = and the value.
This is what the updated code looks like:
fn main() {
let x: felt252 = 10;
println!("x is {}", x);
}
With this change, the code compiles and outputs the value of x
, which in this case is 10
.
For more information on felt252
, you can refer to the Cairo book's Data Types section.
Explanation
- Type Annotation:
let x: felt252;
declares a variablex
of typefelt252
but does not initialize it.- The compiler needs both declaration and initialization.
- Assignment Operator:
- Adding
= 10
initializesx
with the value10
.
- Adding
- Printing the Variable:
println!("x is {}", x);
prints the value ofx
.
This exercise reinforces the importance of initializing variables and understanding type annotations in Cairo. Let's move on to the next exercise.
variables4.cairo
// variables4.cairo
// Execute `starklings hint variables4` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
fn main() {
let x = 3;
println!("x is {}", x);
x = 5; // don't change this line
println!("x is now {}", x);
}
This seems like a pretty simple exercise, and we're told that we're not allowed to change the x = 5
line. But let's look at the errors and see if there's a hint as to how to fix our error.
variables4.cairo
Errors
🟡 Running exercises/variables/variables4.cairo exercise...
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Cannot assign to an immutable variable.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:10:5
x = 5; // don't change this line
^***^
could not compile `exercise_crate` due to previous error
⚠️ Failed to run exercises/variables/variables4.cairo! Please try again.
The key part of the output is: error: Cannot assign to an immutable variable.
If you're familiar with Rust, and now looking at Cairo, we need to remember that variables are immutable by default. This means that we cannot simply assign a new value but must declare the variable that we want to modify as mutable. So let's see how we fix this.
variables4.cairo
Solution
In Cairo, variables are immutable by default, similar to Rust. This means that you cannot reassign a value to a variable unless it is declared as mutable using the mut
keyword.
In this exercise, we are trying to assign x
to 5
after we've already assigned x
to 3
on line 8. This won't work unless we put the mut
keyword in front of x
.
Here's the updated code:
fn main() {
let mut x = 3; // add `mut` keyword
println!("x is {}", x);
x = 5; // don't change this line
println!("x is now {}", x);
}
With this simple update we can now see that our code compiles and we successfully update the value of x
.
Explanation
- Immutability by Default:
- In Cairo, as in Rust, variables are immutable by default. This means that once a variable is assigned a value, it cannot be changed unless explicitly declared as mutable.
- Mutable Variables:
- To make a variable mutable, you need to use the
mut
keyword. This allows you to reassign values to the variable.
- To make a variable mutable, you need to use the
- Assignment and Reassignment:
- The line
let x = 3;
declares an immutable variablex
and assigns it the value3
. - Trying to reassign
x
to5
(x = 5;
) without makingx
mutable results in a compilation error. - Adding
mut
(let mut x = 3;
) makesx
mutable, allowing reassignment.
- The line
By making this simple update, our code compiles, and we successfully update the value of x
.
For more detailed information on variables and mutability in Cairo, you can refer to the Variables and Mutability chapter in the Cairo book.
variables5.cairo
// variables5.cairo
// Execute `starklings hint variables5` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
fn main() {
let number = 1_u8; // don't change this line
println!("number is {}", number);
number = 3; // don't rename this variable
println!("number is {}", number);
}
In variables5.cairo
, we face a similar exercise to variables4.cairo
, but with an additional challenge: we can't change the line let number = 1_u8
, and we also cannot rename the variable number
on line 10
. So, what can we do? Let's see if the compiler can help.
variables5.cairo
Errors
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Cannot assign to an immutable variable.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:10:5
number = 3; // don't rename this variable
^********^
could not compile `exercise_crate` due to previous error
⚠️ Failed to run exercises/variables/variables5.cairo! Please try again.
This output looks familiar—it's the same as the previous exercise. Since we can't use mut
, we need to find another solution. Experienced Rust programmers might already know what to do here: variable shadowing.
variables5.cairo
Solution
In Cairo (as in Rust), variables are immutable by default, but we can use a technique called shadowing. Shadowing allows us to declare a new variable with the same name as a previous variable, effectively creating a new variable that can hold a different value or type.
Shadowing is done using the let
keyword again with the same variable name. This might seem confusing at first, but it's quite powerful as it allows us to reuse variable names within different scopes or with different types.
Here's the updated code:
fn main() {
let number = 1_u8; // don't change this line
println!("number is {}", number);
let number = 3; // don't rename this variable
println!("number is {}", number);
}
Explanation
- Shadowing:
- Shadowing allows you to declare a new variable with the same name as a previous variable. This new variable can have a different value or type.
let number = 1_u8;
declares the firstnumber
with a type ofu8
and value1
.let number = 3;
declares a newnumber
variable, shadowing the previous one. This newnumber
can have a different type or value.
- Print Statements:
- The first
println!("number is {}", number);
prints the value1
. - The second
println!("number is {}", number);
prints the value3
.
- The first
Why Shadowing?
Shadowing is useful because:
- It allows reusing variable names without mutating the original variable.
- It helps in maintaining cleaner code by avoiding unnecessary variable names.
- It allows changing the type of a variable within a certain scope.
For more information on shadowing, you can refer to the Variables and Mutability chapter in the Cairo book.
Alright let's move on to our final variables
exercise!
variables6.cairo
// variables6.cairo
// Execute `starklings hint variables6` or use the `hint` watch subcommand for a hint.
// I AM NOT DONE
use debug::PrintTrait;
const NUMBER = 3;
const SMALL_NUMBER = 3_u8;
fn main() {
println!("NUMBER is {}", NUMBER);
println!("SMALL_NUMBER is {}", SMALL_NUMBER);
}
We have the introduction of the keyword const
in varaibles5.cairo
other than this, we don't have too much else going on but obviously the code is not compiling. Let's take a look at our errors.
variables6.cairo
Errors
🟡 Running exercises/variables/variables6.cairo exercise...
Compiling exercise_crate v0.1.0 (/Users/desmo/repos/starklings-cairo1/runner-crate/Scarb.toml)
error: Unexpected token, `expected ':' followed by a type.`
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:7:13
const NUMBER = 3;
^
error: Unexpected token, expected ':' followed by a type.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:8:19
const SMALL_NUMBER = 3_u8;
^
error: Unknown type.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:7:14
const NUMBER = 3;
^
error: Unknown type.
--> /Users/desmo/repos/starklings-cairo1/runner-crate/src/lib.cairo:8:20
const SMALL_NUMBER = 3_u8;
^
could not compile `exercise_crate` due to previous error
⚠️ Failed to run exercises/variables/variables6.cairo! Please try again.
It seems that the compiler is not happy with how we are declaring our constants and it actually tells us what it is expecting: expected ':' followed by a type.
That's handy. Constants behave a little differently than variables, so let's explore how in our solution below.
variables6.cairo
Solution
Constants are similar to variables in that they are immutable, but they are always immutable. You are not allowed to use the mut
keyword with them. Instead of using let
, you use const
and you must annotate the type of the value.
Knowing this information, let's fix our code. You can read more about this in the Cairo Book here: Constants.
const NUMBER:u8 = 3_u8;
const SMALL_NUMBER:u8 = 3_u8;
fn main() {
println!("NUMBER is {}", NUMBER);
println!("SMALL_NUMBER is {}", SMALL_NUMBER);
}
With these changes, our code compiles successfully and outputs the values of NUMBER
and SMALL_NUMBER
.
Explanation
- Constants Declaration:
- Unlike variables, constants must have their types explicitly annotated.
- The type annotation ensures clarity and helps the compiler enforce strong typing principles.
- Correcting the Code:
const NUMBER: u8 = 3_u8;
declares a constantNUMBER
with typeu8
and value3
.const SMALL_NUMBER: u8 = 3_u8;
declares a constantSMALL_NUMBER
with typeu8
and value3
.
- Print Statements:
- The
println!
macros print the values of the constants.
- The
[!Note] While the type suffix (
3_u8
) provides a hint to the compiler about the type of the literal, Cairo still requires an explicit type annotation for constants to ensure clarity and to uphold the language's strong typing principles.
Conclusion
Congratulations on completing the variables
exercises! By now, you should have a solid understanding of variable declaration, mutability, and constants in Cairo. Here's a quick recap of what we've covered:
-
Variable Declaration and Initialization:
- Learned how to declare and initialize variables.
- Understood the importance of initializing variables to avoid compilation errors.
-
Mutability:
- Explored the concept of immutability by default and how to declare mutable variables using the
mut
keyword.
- Explored the concept of immutability by default and how to declare mutable variables using the
-
Shadowing:
- Discovered how to use variable shadowing to redeclare variables with the same name within different scopes or with different types.
-
Constants:
- Understood the distinction between variables and constants.
- Learned how to declare constants with explicit type annotations and the importance of their immutability.
These exercises have provided a foundational understanding of how Cairo handles variables and constants, which is crucial for more advanced topics and projects. We'll see you in the next set of exercises where we will cover Primitive Types, until then, happy coding!