Macros are more powerful than functions because they expand to produce more code than the code you’ve written manually. For example, a function signature must declare the number and type of parameters the function has. Macros, on the other hand, can take a variable number of parameters: we can call println!("hello") with one argument or println!("hello {}", name) with two arguments. Also, macros are expanded before the compiler interprets the meaning of the code, so a macro can, for example, implement a trait on a given type. A function can’t, because it gets called at runtime and a trait needs to be implemented at compile time.
rust
macro_rules! my_macro {
() => {
println!("Check out my macro!");
};
($val:expr) => {
println!("Look at this other macro: {}", $val);
}
}
fnmain() {
my_macro!();
my_macro!(7777);
}
// Export a macro from a modulemod macros {
#[macro_export]macro_rules! my_macro {
() => {
println!("Check out my macro!");
};
}
}
#![allow(unused)]fnmain() {
// Iterate through a vectorlet my_fav_fruits = vec!["banana", "raspberry"];
letmut my_iterable_fav_fruits = my_fav_fruits.iter();
assert_eq!(my_iterable_fav_fruits.next(), Some(&"banana"));
assert_eq!(my_iterable_fav_fruits.next(), Some(&"raspberry"));
assert_eq!(my_iterable_fav_fruits.next(), None); // When it's over, it's none// One line iteration with action
my_fav_fruits.iter().map(|x| capitalize_first(x)).collect()
// Hashmap iterationfor (key, hashvalue) in &*map {
for key in map.keys() {
for value in map.values() {
}
#![allow(unused)]fnmain() {
let n = 5;
if n < 0 {
print!("{} is negative", n);
} elseif n > 0 {
print!("{} is positive", n);
} else {
print!("{} is zero", n);
}
}
#![allow(unused)]fnmain() {
match number {
// Match a single value1 => println!("One!"),
// Match several values2 | 3 | 5 | 7 | 11 => println!("This is a prime"),
// TODO ^ Try adding 13 to the list of prime values// Match an inclusive range13..=19 => println!("A teen"),
// Handle the rest of cases
_ => println!("Ain't special"),
}
let boolean = true;
// Match is an expression toolet binary = match boolean {
// The arms of a match must cover all the possible valuesfalse => 0,
true => 1,
// TODO ^ Try commenting out one of these arms
};
}
#![allow(unused)]fnmain() {
for n in1..101 {
if n % 15 == 0 {
println!("fizzbuzz");
} else {
println!("{}", n);
}
}
// Use "..=" to make inclusive both endsfor n in1..=100 {
if n % 15 == 0 {
println!("fizzbuzz");
} elseif n % 3 == 0 {
println!("fizz");
} elseif n % 5 == 0 {
println!("buzz");
} else {
println!("{}", n);
}
}
// ITERATIONSlet names = vec!["Bob", "Frank", "Ferris"];
//iter - Doesn't consume the collectionfor name in names.iter() {
match name {
&"Ferris" => println!("There is a rustacean among us!"),
_ => println!("Hello {}", name),
}
}
//into_iter - COnsumes the collectionfor name in names.into_iter() {
match name {
"Ferris" => println!("There is a rustacean among us!"),
_ => println!("Hello {}", name),
}
}
//iter_mut - This mutably borrows each element of the collectionfor name in names.iter_mut() {
*name = match name {
&mut"Ferris" => "There is a rustacean among us!",
_ => "Hello",
}
}
}
An Arc can use Clone to create more references over the object to pass them to the threads. When the last reference pointer to a value is out of scope, the variable is dropped.
rust
#![allow(unused)]fnmain() {
use std::sync::Arc;
let apple = Arc::new("the same apple");
for _ in0..10 {
let apple = Arc::clone(&apple);
thread::spawn(move || {
println!("{:?}", apple);
});
}
}