Rust优于C++的两个原因 ?

开发 前端
运行时类型信息(RTTI),RTTI意味着在运行时执行动态类型检查和类型转换。当启用RTTI时,编译器在二进制文件中包含额外的元数据以支持动态类型信息。

尽管学习曲线陡峭,Rust已经证明了自己是一门值得掌握的语言。今天,让我们深入研究一下Rust优于C++语言的原因。

原因1:积极的编译器优化

  • Rust的编译器(LLVM)比C++编译器更积极地优化代码,这是因为所有权规则,LLVM可以做出假设。
  • LLVM在内联函数方面更加积极,特别是对于小函数。内联避免了函数调用开销并使其快速。

例如,下面的函数可能会或可能不会被C++编译器内联,但LLVM肯定会内联它。

fn f(n: i32, dp: &mut Vec<i32>) -> i32 {
    let n1 = n as usize;
    if dp[n1] != -1 {
        return dp[n1];
    }
    dp[n1] = Self::f(n-1, dp) + Self::f(n-2, dp) + Self::f(n-3, dp);
    dp[n1]
}

原因2:较低的运行时开销

1,C++栈展开导致运行缓慢

什么是栈展开?

每当抛出异常时,在栈上开始分配资源和调用对象的析构函数的过程,这称为栈展开。

class Resource {
public:
    Resource() {
        std::cout << "Resource acquired\n";
    }

    ~Resource() {
        std::cout << "Resource released\n";
    }
};

void foo() {
    Resource res; // Resource acquired
    throw std::runtime_error("Error in foo");
}

int main() {
    try {
        foo();
    } catch (const std::runtime_error& e) {
        std::cerr << "Caught exception: " << e.what() << std::endl;
    }
    return 0;
}

栈展开是如何工作的?

  • 调用foo()时,它获取一个Resource对象。
  • 然后抛出std::runtime_error异常。
  • 作为结果,栈开始展开,并调用res的析构函数来释放Resource。
  • 然后,在main()函数中捕获异常。
  • 这确保了即使在出现异常的情况下也能正确地清理资源。

栈展开有运行时开销,当存在深度嵌套的函数调用或具有复杂析构函数的对象时,将花费时间来释放对象。

2,Rust使用Result和Option类型删除了的栈展开

Rust的Result和Option类型用于错误处理,通过模式匹配而不是异常来处理。

fn divide(a: i32, b: i32) -> Result {
    if b == 0 {
        return Err("Division by zero");
    }
    Ok(a / b)
}

fn main() {
    match divide(10, 0) {
        Ok(result) => println!("Result: {}", result),
        Err(e) => eprintln!("Error: {}", e),
    }
}

3,C++运行时类型信息(RTTI)增加了二进制大小和运行时开销

增加二进制大小:

运行时类型信息(RTTI),RTTI意味着在运行时执行动态类型检查和类型转换。当启用RTTI时,编译器在二进制文件中包含额外的元数据以支持动态类型信息。

这些元数据通常包括:类型信息表(类型描述符)、用于动态调度等的虚函数表(vtable)。这些表增加了二进制文件的大小,特别是对于具有大量多态类的程序。

增加执行时间:

动态强制转换(dynamic_cast),这包括运行时类型检查,以确保转换的正确性。这种类型检查增加了程序执行时间的开销。

虚函数调用,C++语言中的动态多态性适用于虚函数调用,这需要在运行时查找适当的函数。与静态调度相比,会产生额外的运行时开销。

#include <iostream>
#include <typeinfo>

class Base {
public:
    virtual ~Base() {}
};

class Derived : public Base {};

int main() {
    Base* ptr = new Derived();
    Derived* derived = dynamic_cast(ptr);
    if (derived) {
        std::cout << "Dynamic cast successful\n";
    } else {
        std::cout << "Dynamic cast failed\n";
    }
    delete ptr;
    return 0;
}

4,Rust中没有RTTI ???

Rust的类型系统支持多态行为和动态分派(基于trait和enum),而不需要RTTI。Box启用动态分派,不需要运行时类型信息。

trait Printable {
    fn print(&self);
}

struct Base;
struct Derived;

impl Printable for Base {
    fn print(&self) {
        println!("Base");
    }
}

impl Printable for Derived {
    fn print(&self) {
        println!("Derived");
    }
}

fn main() {
    let base: Box = Box::new(Derived);
    base.print();
}

Rust的编译器建立在LLVM上,将高级结构转换为高效的机器码。

Rust优于C++的这两个原因成立吗???

责任编辑:武晓燕 来源: coding到灯火阑珊
相关推荐

2021-12-06 23:00:36

CC++编程语言

2011-07-21 17:11:09

AjaxJSF

2024-02-26 07:26:27

RustC++开发

2009-06-29 10:09:55

AjaxJSF

2023-03-01 10:20:13

2021-06-16 12:03:49

WindowsLinux游戏

2023-09-21 23:35:19

C++变量

2022-02-21 23:08:50

Kubernetes集群容器

2009-08-12 16:37:22

C#变量类型转换

2024-04-08 18:55:54

RustC++开发

2022-10-21 18:41:23

RustC++Azure

2022-06-17 09:46:51

Chrome 102Chrome浏览器

2009-07-15 18:29:22

Jython应用

2024-01-22 10:52:45

C++多线程死锁

2022-03-25 09:57:18

C++Rust语言

2010-07-02 12:26:51

LEACH协议

2009-08-06 15:35:34

C# Web Serv

2020-11-13 07:16:09

线程互斥锁死循环

2024-01-19 21:55:57

C++编程代码

2013-05-30 22:47:40

阿里巴巴阿里云昆塔盒子总动员
点赞
收藏

51CTO技术栈公众号