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 since cairo 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:

  1. Identifier not found: The variable x is not declared before it is used.
  2. Invalid left-hand side of assignment: The x needs to be declared properly.
  3. 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

  1. Declaration with let: In Rust and Cairo, variables must be declared before they are used. The let keyword is used for this purpose. It tells the compiler that a new variable is being declared.
let x = 5;
  1. 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

  1. Variable Declaration and Initialization:
    • In the initial code, let x; declares the variable x but does not initialize it with a value, which is why the compiler throws an error.
    • Correcting it to let x = 10; initializes x with the value 10.
  2. Conditional Statement:
    • The if statement checks whether x is equal to 10.
    • If x equals 10, it prints "x is ten!".
    • Otherwise, it prints "x is not ten!".
  3. Experimentation:
    • Changing the value of x to different numbers and recompiling will result in different outputs based on the ifcondition.

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

  1. Type Annotation:
    • let x: felt252; declares a variable x of type felt252 but does not initialize it.
    • The compiler needs both declaration and initialization.
  2. Assignment Operator:
    • Adding = 10 initializes x with the value 10.
  3. Printing the Variable:
    • println!("x is {}", x); prints the value of x.

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

  1. 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.
  2. Mutable Variables:
    • To make a variable mutable, you need to use the mut keyword. This allows you to reassign values to the variable.
  3. Assignment and Reassignment:
    • The line let x = 3; declares an immutable variable x and assigns it the value 3.
    • Trying to reassign x to 5 (x = 5;) without making x mutable results in a compilation error.
    • Adding mut (let mut x = 3;) makes x mutable, allowing reassignment.

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

  1. 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 first number with a type of u8 and value 1.
    • let number = 3; declares a new number variable, shadowing the previous one. This new number can have a different type or value.
  2. Print Statements:
    • The first println!("number is {}", number); prints the value 1.
    • The second println!("number is {}", number); prints the value 3.

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

  1. Constants Declaration:
    • Unlike variables, constants must have their types explicitly annotated.
    • The type annotation ensures clarity and helps the compiler enforce strong typing principles.
  2. Correcting the Code:
    • const NUMBER: u8 = 3_u8; declares a constant NUMBER with type u8 and value 3.
    • const SMALL_NUMBER: u8 = 3_u8; declares a constant SMALL_NUMBER with type u8 and value 3.
  3. Print Statements:
    • The println! macros print the values of the constants.

[!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:

  1. Variable Declaration and Initialization:

    • Learned how to declare and initialize variables.
    • Understood the importance of initializing variables to avoid compilation errors.
  2. Mutability:

    • Explored the concept of immutability by default and how to declare mutable variables using the mut keyword.
  3. Shadowing:

    • Discovered how to use variable shadowing to redeclare variables with the same name within different scopes or with different types.
  4. 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!