Discover millions of ebooks, audiobooks, and so much more with a free trial

From $11.99/month after trial. Cancel anytime.

Beginning Rust: From Novice to Professional
Beginning Rust: From Novice to Professional
Beginning Rust: From Novice to Professional
Ebook597 pages4 hours

Beginning Rust: From Novice to Professional

Rating: 0 out of 5 stars

()

Read preview

About this ebook

Learn to program with Rust in an easy, step-by-step manner on Unix, Linux shell, macOS and the Windows command line.  As you read this book, you’ll build on the knowledge you gained in previous chapters and see what Rust has to offer.  
Beginning Rust starts with the basics of Rust, including how to name objects, control execution flow, and handle primitive types. You’ll see how to do arithmetic, allocate memory, use iterators, and handle input/output. Once you have mastered these core skills, you’ll work on handling errors and using the object-oriented features of Rust to build robust Rust applications in no time.
Only a basic knowledge of programming is required, preferably in C or C++. To understand this book, it's enough to know what integers and floating-point numbers are, and to distinguish identifiers from string literals.

After reading this book, you'll be ready to build Rust applications.  
What You'll Learn
  • Get started programming with Rust
  • Understand heterogeneous data structures and data sequences
  • Define functions, generic functions, structs, and more
  • Work with closures, changeable strings, ranges and slices
  • Use traits and learn about lifetimes  

Who This Book Is For 
Those who are new to Rust and who have at least some prior experience with programming in general: some C/C++ is recommended particularly.

LanguageEnglish
PublisherApress
Release dateMar 22, 2018
ISBN9781484234686
Beginning Rust: From Novice to Professional

Related to Beginning Rust

Related ebooks

Programming For You

View More

Related articles

Reviews for Beginning Rust

Rating: 0 out of 5 stars
0 ratings

0 ratings0 reviews

What did you think?

Tap to rate

Review must be at least 10 words

    Book preview

    Beginning Rust - Carlo Milanesi

    © Carlo Milanesi 2018

    Carlo MilanesiBeginning Rusthttps://doi.org/10.1007/978-1-4842-3468-6_1

    1. Printing on the Terminal

    Carlo Milanesi¹ 

    (1)

    Bergamo, Italy

    In this chapter, you will learn:

    How to write and run your first program in the Rust Language

    How to output texts and numbers on the terminal

    How to write a small script that makes more readable the output of the compiler

    How to write comments in your code

    How to Start

    The smallest valid Rust program is:

    fn main(){}

    Of course, it does nothing. It just defines an empty function named main. By function we mean a set of instructions that does something and that has been given a name.

    The fn word is shorthand for function. The main word is the name of the function . The round parentheses contain the possible arguments of the function; in this case there are no arguments. To close, the braces contain the possible statements that comprise the body of the function; in this case there is no statement.

    When a program written in Rust is run, its main function is executed. If there is no main function, then it isn’t a complete program; it may be a library, though.

    To run this program, complete the following actions:

    Install the package containing the Rust compiler and its utilities. A compiler can be downloaded for free from the website https://www.rust-lang.org. Linux, Windows, or macOS platforms, with processor architecture x86 (32-bit) or x86-64 (64-bit), are available. For each platform, there are three versions: stable, beta, and nightly. The stable version is recommended; it is the oldest, but also the most tested and the one less likely to change. All of these program versions should be used from the command line of a console. After installation, to check which version is installed, type at a command line (with uppercase V): rustc -V.

    Create or choose a folder where your Rust exercises will be stored and, using any text editor, create in that folder a file named main.rs, having the above contents.

    At a command line, in the folder, type: rustc main.rs. The prompt should be printed almost immediately, after having created a file named main (in a Windows environment, it will be named main.exe). Actually, the rustc command has correctly compiled the specified file. That is, it has read it and generated the corresponding machine code, and it has stored this machine code in a file in the same folder.

    At the command line, if in a Windows environment, type: main For other operating systems, type: ./main. You just run the program generated before, but the prompt should be printed immediately, as this program does nothing.

    Hello, World!

    Let's see how to print some text on the terminal. Change the program of the previous section to the following:

    fn main() {

        print!(Hello, world!);

    }

    If it is compiled and run as before, it prints: Hello, world!

    Notice that the newly added line contains eight syntax items, aka tokens. Let’s examine them:

    print: It is the name of a macro defined in the Rust standard library.

    !: It specifies that the preceding name indicates a macro. Without such a symbol, print would instead indicate a function. There is no such function in the Rust standard library, and so you would get a compilation error. A macro is a thing similar to a function - it’s some Rust code to which a name is associated. By using this name, you ask to insert such code in this point.

    (: It starts the list of arguments of the macro.

    ": it starts the literal string.

    Hello, world!: It is the content of the literal string.

    ": It ends the literal string.

    ): It ends the list of the arguments of the macro.

    ;: It ends the statement .

    Let’s examine the meaning of the literal string phrase. The word string means finite sequence of characters, possibly including spaces and punctuation. The word literal means with a value specified directly in source code. Therefore a literal string is a finite sequence of characters (possibly including spaces and punctuation) specified directly in source code.

    The print macro simply inserts some code that prints on the terminal the text that is received as an argument.

    Rust always distinguishes between uppercase and lowercase letters - it’s case sensitive. For all characters that are outside literal strings and comments, if you replace some uppercase letter with a lowercase one or conversely, typically you get a compilation error, or anyway a program with a different behavior. Instead, making such errors inside literal strings always allows a successful compilation, but it's likely the behavior of the program will be different.

    For example:

    fn Main() {}

    If you compile this program, you get the compilation error main function not found, as no main function (with a lowercase m) is defined in the program.

    From now on, except when specified, we will assume the example code will be inside the braces of the main function, and so the braces and the text preceding them will be omitted.

    Printing Combinations of Literal Strings

    Instead of using a single literal string , you can print several of them, even in a single statement. In this way:

    print!({}, {}!, Hello, world);

    This statement, put inside the braces of the main function, will print again: Hello, world!.

    In this case, the print macro receives three arguments, separated by commas. All three arguments are literal strings. The first string, though, contains two pairs of braces ({}). They are placeholders, indicating the positions in which to insert the other two strings.

    So, the macro scans the arguments after the first one, and for each of them it looks inside the first argument for a pair of braces, and replaces them with the current argument.

    This resembles the following C language statement:

    printf(%s, %s!, Hello, world);

    But there is an important difference. If you try to compile

    print!({}, !, Hello, world);

    you get the compilation error argument never used, as the arguments of the macro after the first one are more numerous than the placeholders inside the first argument; that is, there is some argument that doesn't have any corresponding placeholder.

    And if you try to compile

    print!({}, {}!, Hello);

    you get a compilation error too, as the placeholders inside the first argument are more numerous than the arguments after the first argument; that is, there is a placeholder that doesn't have any corresponding argument.

    Instead, the corresponding statements in the C language do not raise compilation errors , but instead cause the compiled program to crash or misbehave.

    Printing Several Lines of Text

    So far, we wrote programs that print only one line. But a single statement can print several lines; in this way:

    print!(First line\nSecond line\nThird line\n);

    This will print:

    First line

    Second line

    Third line

    The sequence of characters \n, where n stands for new line, is transformed by the compiler into the character sequence that represents the line terminator for the currently used operating system.

    Given that it is very common to go to a new line only once for every printing statement, and just at the end of the statement, another macro, println, has been added to the Rust standard library. It’s used in this way:

    println!(text of the line);

    This statement is equivalent to:

    print!(text of the line\n);

    Calling this println macro (whose name is to be read print line) is equivalent to calling print with the same arguments, and then outputting a line terminator.

    Printing Integer Numbers

    If we want to print an integer number , we can type:

    print!(My number: 140);

    or, using a placeholder and an additional argument:

    print!(My number: {}, 140);

    or, removing the quotes around the second argument:

    print!(My number: {}, 140);

    All these statements will print: My number: 140.

    In the last statement, the second argument is not a literal string, but it is a literal integer number, or, for short, a literal integer.

    The integers are another data type, with respect to strings.

    Even for integers, the print macro is able to use them to replace the corresponding placeholder inside its first argument.

    In fact, the compiler interprets the string 140 contained in the source code as a number expressed in decimal format, it generates the equivalent number in binary format, and then it saves it into the executable program .

    At runtime, the program takes such number in binary format, it transforms it into the string 140, using the decimal notation, then it replaces the placeholder with that string, so generating the string to print, and finally it sends the string to the terminal.

    This procedure explains, for example, why if the following program is written:

    print!(My number: {}, 000140);

    the compiler generates exactly the same executable program generated before. Actually, when the source string 000140 is converted to the binary format, the leading zeros are ignored.

    The argument types may be mixed too. This statement

    print!({}: {}, My number, 140);

    will print the same line as before. Here the first placeholder corresponds to a literal string, while the second one to a literal integer.

    Command-Line Script

    The rustc command shown above has a drawback: it prints all the errors it finds in your code in the order in which it finds them. Well, often your code will contain many syntax errors, and you should process them from the first one to the last one. But after the compiler has finished printing the errors, you are faced with the new prompt preceded by the last error found. So, you have to scroll back to the first error.

    A way to improve this situation is to use a command-line script, whose syntax depends on the operating system.

    In a Linux system, you can put the following lines in a new script file:

    clear

    rustc $* --color always 2>&1 | more

    In a Windows system, you can put the following three lines in a .BAT file:

    @echo off

    cls

    rustc %* --color always 2>&1 | more

    If the script file is named, say, rs (rs.bat on Windows), to compile a file named main.rs you can type:

    rs main.rs

    This script first clears the screen, then runs the rustc compiler with all the arguments you give to it. If the compilation is successful, or if it generates less than a screenful of error messages, it behaves like a normal run of rustc.

    Otherwise, it fills the screen with error messages, and then stops, and shows the message --More-- at the bottom of the terminal screen. At this point you can press:

    The Enter key, to advance by one line.

    The Space key, to advance by one screenful.

    The Q key (for quit), to abort printing error messages , and get back to the command prompt.

    Comments

    Write the following code

    // This program

    // prints a number.

    print!({}, 34); // thirty-four

    /* print!({}, 80);

    */

    that will print 34.

    The first two lines start with a pair of slashes //. Such a pair of characters indicates the start of a line comment, ending at the end of the line. To write a comment on several lines, the pair of slashes must be repeated at every line of the comment, as in the second line of the program above.

    Rust programmers use to leave a blank just after the double slash, to improve readability.

    As it appears in the third line, a line comment may start after a statement, usually separated by at least one blank.

    There is another kind of comment, exemplified in the fourth and fifth lines. Such a comment starts with the character pair /* and ends with the pair */; it may extend on several lines, and so it is named multi-line comment.

    Rust programmers usually avoid the multi-line comment in production code, using only single line comments, and use multi-line comments only to exclude temporarily some code from compilation.

    Rust comments look identical to modern C language comments. In fact, there is an important difference between Rust comments and C comments: Rust comments may be nested, and they must be nested correctly.

    /* This is /* a valid*/

    comment, even /* if /* it contains

    comments*/ inside */itself.  */

    /* This /* instead is not allowed in Rust,

    while in C is tolerated (but it may generate a warning).*/

    © Carlo Milanesi 2018

    Carlo MilanesiBeginning Rusthttps://doi.org/10.1007/978-1-4842-3468-6_2

    2. Doing Arithmetic

    Carlo Milanesi¹ 

    (1)

    Bergamo, Italy

    In this chapter, you will learn:

    How to compute an arithmetic operation between integer numbers or between floating-point numbers

    How to write a program containing several statements

    How to print strings in several lines

    Adding Integer Numbers

    Let’s see how to compute the sum of two integer numbers ; for example, 80 and 34.

    Put the following line as the only contents of the braces of the main function:

    print!(The sum is {}., 80 + 34);

    The execution will print: The sum is 114..

    The second argument of the print macro is the expression 80 + 34.

    The compiler surely does not store in the executable such numbers in decimal format. If compiling optimizations are off, the compiler just converts the two numbers in binary format and stores into the executable such binary numbers and the addition operation. But if compiling optimizations are on, the compiler, realizing that this expression contains only constant values, evaluates directly that expression, obtaining the integer number 114, and stores into the executable program such number in binary format. In both cases, at runtime, such a number is formatted as the three-character decimal string 114, and then the placeholder {} of the literal string is replaced by such string. Finally, of course, the resulting string is printed to the console.

    Notice that the string The sum is 114. has been generated by the program; it is not present in source code; so it is still a string, but not a literal string.

    Similarly, the two-character sequence 80 represents an integer number directly in source code, and so it is called a literal integer. The same holds for the two characters 34. Instead, the integer number 114, saved into the executable in binary format, and loaded into memory at runtime, is not a literal integer, as it does not appear in source code.

    It is also allowed to write:

    print!({} + {} = {}, 34, 80, 80 + 34);

    whose execution will print 34 + 80 = 114.

    In such case, the second argument of the macro will be put where there is the first placeholder, the third argument where there is the second placeholder, and the fourth argument where there is the third placeholder.

    You can specify hundreds of arguments for the print macro, as long as the arguments after the first one are as many as the placeholders {} inside the first argument.

    Other Operations Between Integer Numbers

    All the integer arithmetic operators of C language can be used. For example:

    print!({}, (23 - 6) % 5 + 20 * 30 / (3 + 4));

    This will print 87.

    Let’s see why.

    Such a formula is evaluated by the Rust compiler exactly as the C compiler would.

    First, the operations in parentheses, 23 - 6 and 3 + 4, are evaluated, obtaining, respectively, 17 and 7.

    At this point, our expression has become 17 % 5 + 20 * 30 / 7.

    Then, the multiplication and division operations are evaluated, as they have precedence over addition and subtraction, and, for operations of the same precedence, they are evaluated in order from left to right.

    17 % 5 is the remainder of the integer division operation, and it has 2 as a result, that is, the remainder of the operation 17 / 5. The expression 20 * 30 is evaluated before the following division, as it is at its left.

    At this point, our expression has become 2 + 600 / 7.

    Then, the integer division (with truncation) 600 / 7 is performed, and our expression has become 2 + 85.

    Finally, the sum is evaluated, the result is formatted in decimal notation, the placeholder is replaced by the formatted number, and the resulting string is printed.

    These arithmetic operations are performed always on integer binary numbers, obtaining integer binary numbers, and the results are converted to decimal format only when they are used to replace the placeholders.

    Usually, compilers generate machine language instructions, which are afterwards executed when the program is running; though, the Rust compiler is highly optimizing, and so it tries to evaluate directly at compile time the expressions that it is possible to evaluate using the information available in source code. Our expression is made only of literal integers, and so the whole expression will be evaluated already at compile time, storing in the executable program only the result to print. Though, conceptually, we can think that the computations are performed at runtime.

    Floating-Point Arithmetic

    Let's see how to compute the sum between two numbers with fractional parts, for example, 80.3 and 34.9.

    Replace the statement with:

    print!(The sum is {}, 80.3 + 34.8);

    The run will print The sum is 115.1.

    Now replace in the second number only the character 8 with a 9, obtaining:

    print!(The sum is {}, 80.3 + 34.9);

    The run will print The sum is 115.19999999999999.

    This will surprise those who would expect 115.2 as the result.

    This phenomenon happens also in many other programming languages, and it is due to the fact that Rust, like almost every programming language, performs computations involving non-integer numbers using the floating-point format. But here we won’t treat this format any more.

    Also expressions containing floating-point numbers can be evaluated:

    print!({}, (23. - 6.) % 5. + 20. * 30. / (3. + 4.));

    This will print 87.71428571428571.

    Let's see why.

    By putting a dot after a literal number, it is transformed into a literal floating-point number, having the same value. The precedence rules are the same of integer arithmetic, although division has a different result.

    Let's see how the evaluation of the expression is performed.

    The evaluation of 23. - 6. and of 3. + 4. is similar to that of integer numbers.

    By evaluating 17. % 5., 2. is obtained, similarly to integer numbers. Such an operator does not exist in C language for floating-point numbers, and it corresponds to the expression fmod(17., 5.) of the C language.

    By evaluating 20. * 30., 600. is obtained, similarly to integer numbers.

    By evaluating 600. / 7., a floating-point number is obtained, that cannot be exactly represented neither in binary notation nor in the decimal one. Internally, a binary-format approximate representation is generated; if you ask Rust to convert such number into a decimal-format approximate representation, you would get the string 85.71428571428571.

    Finally, the value 2. is added to such binary number , obtaining another value that cannot be exactly represented, which is printed in the way shown above.

    Notice that, differing from C language, in Rust you cannot simply mix integer numbers and floating-point numbers. The following statement generates a compilation error:

    print!({}, 2.7 + 1);

    A way to make it valid is to add a dot:

    print!({}, 2.7 + 1.);

    However, this one is a syntax-only limitation, not an operative one, as anyway machine code cannot sum an integer number and a floating-point number, without before converting one of the two operands to the type of the other operand. A C compiler, when it encounters the expression 2.7 + 1, emits implicitly the machine language instruction to convert the integer number 1 to a floating-point number, or better, being 1 a constant, it is converted to a floating-point number at compile time. In Rust, such conversions must be explicit.

    At last, a note about the % operator. This is often improperly named modulo (or modulus) operator. Well, it should be better named remainder operator, because the mathematical modulo operator has a different behavior for negative numbers. The % operator behaves in Rust like in C language:

    print!({} {}, -12 % 10, -1.2 % 1.);

    This will print -2 -0.19999999999999996.

    Sequences of Statements

    As the body of the main function, write:

    print!({} + , 80);

    print!({} =, 34);

    print!( {}, 80 + 34);

    This will print 80 + 34 = 114.

    The program now contains three statements , each of them terminated by the ; character. Such statements are executed in order of appearance.

    If the body of the main function would become

        print!({} + ,80);print!({} = ,34);

                print  ! ( {}  ,

            80     + 34 )  ;

    its result wouldn't change. Actually, additional whitespaces (blanks, tabs, and line breaks) are ignored.

    However, Rust programmers have the following habits that are recommended:

    to indent lines by four spaces inside functions;

    to avoid adding several consecutive spaces inside statements;

    to avoid exceeding 80 columns, possibly splitting long statements on several lines.

    Breaking Literal Strings

    As said before, to avoid code lines that are too long, you can break them in any point between syntax symbols, like in C language; though, the syntax to break a literal string is different. This code is illegal:

    println!({}, This

        is not allowed);

    Actually, in Rust you cannot simply juxtapose literal strings, like in C.

    Though, you can start a literal string in one line, and end it a few lines below. For example, this is a valid program:

    fn main() {

        println!({}, "These

            are

            three lines");

    }

    And this is what is printed:

    These

            are

            three lines

    As you can see, the literal string contains all the characters that in the source file are between the start and the end of the string, including newline characters and line leading spaces.

    Maybe this is what you want, but you can do something different:

    fn main() {

        println!({}, "This \

            is \

            just one line");

    }

    This will print:

    This is just one line

    By adding a backslash character ("\") inside a literal string, just before the end of a line, the resulting string will contain neither that end-of-line character nor the following spaces; therefore, the leading spaces of the next line are omitted. Given that we wanted at least one blank, we inserted such a blank just before the backslashes.

    Finally, if we want a single literal string containing several resulting lines, with no leading whitespace, we can write this:

    fn main() {

        println!({}, "These

    are

    three lines");

    }

    or this:

    fn main() {

    println!({}, "These\n\

        are\n\

        three lines");

    }

    Both will print:

    These

    are

    three lines

    The first solution has the drawback of being disrespectful of indentation conventions, and therefore usually the second solution is preferable. In such a solution, at the end of the lines there is the sequence \n, which is codified as a newline sequence, and then another backslash to exclude from the string the source code newline and the following spaces.

    © Carlo Milanesi 2018

    Carlo MilanesiBeginning Rusthttps://doi.org/10.1007/978-1-4842-3468-6_3

    3. Naming Objects

    Carlo Milanesi¹ 

    (1)

    Bergamo, Italy

    In this chapter, you will learn:

    The concepts of value, object, and variable

    The concept of mutability of variables

    The difference between initialization and re-assignment

    How to avoid warnings for unused variables

    The concept of Boolean expression

    Which type checks are performed by the compiler for assignments

    How some operators can perform both an arithmetic operation and an assignment

    How to invoke functions defined in the Rust standard library

    Associating Names to Values

    So far, we have seen three kinds of values: strings, integer numbers, and floating-point numbers .

    But values should not be confused with objects and variables. So, let’s define what the words value, object, and variable actually mean.

    The word value indicates an abstract, mathematical concept. For example, when you say the value 12 you mean the mathematical concept of the number 12. In mathematics, there is just one number 12 in the world. Even true or Hello are values that conceptually exist in one single instance in the universe, because they are concepts.

    But values may be stored in the memory of a computer. You can store the number 12 or the string Hello in several locations of the memory. So, you can have two distinct memory locations that both contain the 12 value.

    The portion of memory that contains a value is named object. Two distinct objects, located in different positions of memory, are said to be equal if they contain the same value. Instead, two values are said to be equal if and only if they are not distinct, that is, they are actually the same value.

    When Rust source code is compiled, the resulting executable program contains only objects that have a memory location and a value. Such objects do not have names. But in source code, you may want to associate names to objects, to reference them later. For example, you can write, as the body of the main function:

    let number = 12;

    let other_number = 53;

    print!({}, number + other_number);

    This will print 65.

    The word let, as the already seen word fn, is a keyword reserved by the language, that is, a word that cannot be used for other purposes.

    The first statement introduces in the program an object containing the value 12 and associates the name number to such object. The second statement introduces another object and associates it to another name. The third statement accesses these two objects by using the names previously defined.

    The first statement has the following effects:

    it reserves an object (i.e., an area of memory) large enough to contain an integer number;

    it stores the value 12 in such object, in binary format;

    it associates the name number to such object, so that such name can be used in later points of the source code to indicate such object.

    Therefore, such a statement is not a simple alias declaration. It doesn’t mean From now on, every time we use the word number, we will mean the value 12; rather, it means A memory space shall be reserved to contain initially the value 12, and from now on, every time we use the word number, we will mean such memory space. So, such statement declares both an object and a name of that object. A synonym of name of object is identifier.

    Identifier-object pairs are called variables . So, that statement is a declaration of a variable.

    But it is more than a declaration. A simple declaration of a variable just reserves space for an object, and associates an identifier to such object. The value of the object remains undefined. Instead, this statement also assigns the initial value of such object. Assigning the initial value of an object is called initializing that object. So, we say that this statement declares and initializes a variable.

    The operation of reserving a memory area for an object is named allocation of that object. Conversely, the removal of an object, causing its memory area to become available for allocating other objects, is named deallocation of that object. So we say that this statement allocates an object, assigns it to an identifier, and initializes that object (or the variable composed by the pair identifier-object).

    These concepts are the same as those of C language.

    The second statement is similar to the first one.

    After having declared and initialized a variable, you can use the name of such variable inside expressions, and the evaluation of such variable gives the value stored in its object. Actually, the third statement above appears to add the names of the two variables, with the effect of adding their current values.

    If any of the first two statements were omitted, the third statement would generate a compilation error, as it would use an undeclared variable; and this is forbidden.

    In the second statement of the following code

    let number = 12;

    print!({} {}, number, 47);

    two numbers are printed , 12 and 47, but 12 is printed as it is the value of a variable, while 47 is a literal.

    Mutable Variables

    After having appropriately declared a variable , you can modify its value in another kind of statement, named assignment:

    let mut number = 12;

    print!({}, number);

    number = 53;

    print!( {}, number);

    This will print 12 53.

    The first statement declares the variable number and initializes it to the value 12. The second statement prints the value of such variable. The third statement assigns the value 53 to the same variable. The fourth statement prints the new value of the variable.

    The assignment does not allocate objects. It just modifies the value of an already allocated object.

    You probably noticed that the first statement contains the word mut; it is a keyword of Rust, and it is an abbreviation of mutable. In fact, the conventional name variable is somewhat improper, as it

    Enjoying the preview?
    Page 1 of 1