自学内容网 自学内容网

rust学习-函数的定义与使用

1. 函数的基本定义

在Rust中,函数使用fn关键字来定义:

fn say_hello() {
    println!("Hello, world!");
}
  • fn:定义函数的关键字
  • say_hello:函数的名字
  • ():函数的参数列表,当前函数没有参数
  • {}:函数体,包含了函数执行的代码

2. 函数的参数

函数可以接受参数,参数的类型必须在函数签名中明确指定:

fn greet(name: &str) {
    println!("Hello, {}!", name);
}
  • name: &str:函数的参数定义,name是参数的名称,&str是参数的类型(&str表示一个字符串切片)

多个参数

函数可以有多个参数,参数之间用逗号分隔:

fn add(x: i32, y: i32) {
    println!("The sum of {} and {} is {}.", x, y, x + y);
}
  • x: i32, y: i32:函数有两个参数,x和y,它们的类型都是i32(32位整数)

3. 返回值

函数可以返回一个值,返回值的类型在函数签名中使用->符号指定:

fn add(x: i32, y: i32) -> i32 {
    x + y
}
  • -> i32:表示函数返回一个i32类型的值
  • x + y:函数的返回值,在Rust中,函数体中的最后一个表达式的值会自动作为返回值,除非使用return关键字提前返回

提前返回

可使用return关键字从函数中间返回一个值:

fn divide(x: i32, y: i32) -> Option<i32> {
    if y == 0 {
        return None; // 提前返回
    }
    Some(x / y)
}
  • Option:表示函数的返回值是一个Option类型,它可以是Some(i32)或None
  • return None;:在y == 0的情况下,函数会提前返回None

4. 函数调用

定义函数后,可以通过函数名来调用它,调用函数时,必须提供与函数签名中定义的参数相匹配的参数:

fn main() {
    say_hello(); // 调用无参数函数
    greet("XiaoMing"); // 调用带有一个参数的函数
    add(3, 4); // 调用带有两个参数的函数

    let result = add(5, 6); // 调用带有返回值的函数
    println!("The result is {}.", result);
}
  • say_hello():调用无参数的函数
  • greet(“XiaoMing”):调用带有一个参数的函数,传入字符串"XiaoMing"
  • add(3, 4):调用带有两个参数的函数,传入3和4
  • let result = add(5, 6):调用带有返回值的函数,并将返回值存储在变量result中

5. 函数的所有权和借用

Rust的所有权系统对函数参数的处理有一些特殊的规则,当将一个变量传递给函数时,Rust会根据参数的类型来决定是传递所有权还是借用

传递所有权

如果参数类型是T(非引用类型),则传递所有权,意味着函数内部对该参数的修改不会影响外部变量,因为外部变量的所有权已经被转移给函数了,例如:

fn take_ownership(s: String) {
    println!("Inside function: {}", s);
} // 这里s离开作用域,内存被释放

fn main() {
    let my_string = String::from("Hello");
    take_ownership(my_string); // my_string的所有权被转移
    // println!("{}", my_string); // 这里会报错,因为my_string的所有权已被转移
}

借用

如果参数类型是&T(引用类型),则传递的是引用(借用),意味着函数内部对参数的修改不会影响外部变量,且函数执行完毕后,外部变量仍然拥有所有权,例如:

fn borrow(s: &String) {
    println!("Inside function: {}", s);
} // 这里s离开作用域,但不会释放内存,因为只是借用

fn main() {
    let my_string = String::from("Hello");
    borrow(&my_string); // 传递借用
    println!("{}", my_string); // 这里可以正常打印my_string
}

6. 函数作为参数和返回值

Rust支持将函数作为参数传递给其他函数,也可将函数作为返回值返回,这需要使用函数指针(fn)或闭包(Fn trait)

函数作为参数

可以将函数作为参数传递给另一个函数:

fn apply(f: fn(i32) -> i32, x: i32) -> i32 {
    f(x)
}

fn square(x: i32) -> i32 {
    x * x
}

fn main() {
    let result = apply(square, 5); // 将square函数作为参数传递
    println!("Result: {}", result); // 输出25
}
  • apply函数接受一个函数f作为参数,并调用它
  • square函数计算一个数的平方,并返回结果

函数作为返回值

可以返回一个函数:

fn create_greeter(greeting: &str) -> impl Fn(&str) -> String {
    move |name| format!("{} {}!", greeting, name)
}

fn main() {
    let greeter = create_greeter("Hello");
    println!("{}", greeter("XiaoMing")); // 输出 "Hello XiaoMing!"
}
  • create_greeter函数返回一个闭包(匿名函数),这个闭包接受一个字符串参数并返回一个新的字符串
  • greeter变量存储了返回的闭包,并在调用时使用

7. 泛型函数

Rust支持泛型函数,可编写可以处理多种类型的函数,泛型函数使用语法来定义泛型类型参数:

fn print_twice<T: std::fmt::Debug>(value: T) {
    println!("{:?}", value);
    println!("{:?}", value);
}

fn main() {
    print_twice(3); // 输出3两次
    print_twice("Hello"); // 输出"Hello"两次
}
  • <T: std::fmt::Debug>:表示T是一个泛型类型,它必须实现了Debug trait(用于格式化输出)
  • print_twice函数可以接受任何实现了Debug trait的类型

8. 函数注释(文档注释)

Rust支持使用文档注释来为函数添加注释,这些注释可以生成文档,文档注释以///开头:

/// 这个函数接受两个整数并返回它们的和
///
/// # Examples
///
/// ```
/// let result = add(3, 4);
/// assert_eq!(result, 7);
/// ```
fn add(x: i32, y: i32) -> i32 {
    x + y
}
  • ///:表示这是一个文档注释
  • #Examples:用于展示示例代码
  • 这些注释可以通过cargo doc命令生成文档

原文地址:https://blog.csdn.net/weixin_42333247/article/details/145258118

免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!