百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

从C和C++迈向Rust:基础篇

liebian365 2025-02-06 15:35 13 浏览 0 评论

在编程世界中,C和C++凭借其高效、灵活以及对底层硬件的强大操控能力,长期占据着重要地位,被广泛应用于系统开发、游戏开发、嵌入式等诸多领域。而Rust作为一门新兴的系统编程语言,近年来备受瞩目,它以内存安全、零成本抽象、并发性强等特性,在追求高性能与可靠性的场景中崭露头角。对于精通C和C++的开发者而言,借助已有的知识储备,通过类比的方式学习Rust,可以更高效地掌握这门新语言的基础。

一、语法基础

(一)变量声明与赋值

在C和C++中,变量声明需要指定类型,并且可以在声明时进行初始化。例如:

int num = 10;

double pi = 3.14;


在Rust中,变量声明同样需要指定类型(大部分情况下类型可以自动推断),使用 let 关键字。例如:

let num = 10;

let pi: f64 = 3.14;


这里可以看到,Rust的 let 声明更为简洁,且在类型推断机制下,很多时候无需显式写出类型。不过当类型推断不明确时,就需要像 pi 的声明一样显式指定类型,如 f64 表示64位浮点数。另外,Rust中变量默认是不可变的,若要可变,需加上 mut 关键字:

let mut count = 0;

count = 1;


这与C和C++中变量默认可变不同,Rust这种设计有助于减少因意外修改变量导致的错误。

(二)数据类型

C和C++拥有丰富的数据类型,包括基本类型(如 int 、 char 、 float 等)和复合类型(如数组、结构体、联合体等)。例如,定义一个结构体:

struct Point {

int x;

int y;

};


在Rust中,定义结构体的方式为:

struct Point {

x: i32,

y: i32,

}


语法结构相似,但Rust的结构体字段声明使用 : 分隔字段名和类型。此外,Rust的基本类型命名更为规范,如 i32 表示32位有符号整数, u8 表示8位无符号整数等,相比C和C++中 int 在不同平台可能有不同字节数,Rust的类型定义更清晰,可移植性更强。

Rust还引入了枚举类型( enum ),这在C和C++中虽然也有类似概念,但Rust的枚举更为强大。例如:

enum Result {

Success(i32),

Failure(String),

}


这里 Result 枚举可以表示成功(携带一个 i32 类型的值)或失败(携带一个 String 类型的错误信息)两种状态,在处理可能失败的操作时非常实用,而C和C++通常需要自定义结构体来实现类似功能。

(三)函数定义与调用

C和C++中函数定义包括返回类型、函数名、参数列表和函数体。例如:

int add(int a, int b) {

return a + b;

}


调用函数时直接使用函数名和参数: int sum = add(3, 5);

Rust的函数定义语法类似:

fn add(a: i32, b: i32) -> i32 {

a + b

}


调用方式也相似: let sum = add(3, 5); 。不过Rust中函数体最后一个表达式的值会自动作为返回值,无需显式使用 return 关键字(除非提前返回)。

二、内存管理

(一)栈与堆

在C和C++中,基本数据类型和局部变量通常存储在栈上,而通过 new (C++)或 malloc (C)分配的内存则在堆上。例如:

int num = 10; // 栈上

int* ptr = new int(20); // 堆上


Rust同样有栈和堆的概念。栈上存储简单、大小固定的数据,而复杂或大小可变的数据存储在堆上。不过Rust通过所有权系统来管理内存,而不是像C和C++那样依赖手动内存分配和释放。例如:

let num = 10; // 栈上

let v = Vec::new(); // Vec是一个在堆上分配内存的动态数组


这里 num 是简单的整数,存储在栈上,而 Vec 是一个动态数组,其数据存储在堆上,但 v 这个变量本身存储在栈上,它拥有堆上数据的所有权。

(二)所有权与借用

Rust的所有权系统是其核心特性之一,用于确保内存安全。每个值在Rust中都有一个所有者(owner),当所有者离开作用域时,其拥有的值会被自动释放。例如:

{

let s = String::from("hello");

} // s离开作用域,其占用的堆内存被自动释放


在C和C++中,需要手动使用 delete (C++)或 free (C)来释放堆内存,否则会导致内存泄漏。

Rust还引入了借用(borrowing)的概念,允许在不转移所有权的情况下使用值。例如:

fn print_length(s: &String) {

println!("Length: {}", s.len());

}


let s = String::from("world");

print_length(&s); // 借用s,不转移所有权


这里 &String 表示对 String 的引用, print_length 函数可以通过这个引用访问 String 的值,但不拥有其所有权。这类似于C和C++中的指针,但Rust的借用检查器会在编译时确保引用的安全性,避免空指针、野指针等问题。

(三)智能指针

C++中有智能指针(如 std::unique_ptr 、 std::shared_ptr 、 std::weak_ptr )来帮助管理内存,避免手动内存管理的复杂性。Rust也有类似的概念,如 Box 、 Rc (引用计数)、 Arc (原子引用计数,用于多线程环境)等。

Box 类似于C++的 std::unique_ptr ,用于在堆上分配单个值,并在其离开作用域时自动释放。例如:

let b = Box::new(10);


Rc 类似于C++的 std::shared_ptr ,通过引用计数来管理堆上的值,允许多个所有者共享同一个值。例如:

use std::rc::Rc;


let a = Rc::new(10);

let b = a.clone();


这里 a 和 b 共享堆上的值 10 ,当引用计数为0时,值被释放。而 Arc 用于多线程环境,提供原子操作来保证引用计数的线程安全。

三、控制流

(一)条件语句

C和C++的条件语句主要是 if - else 和 switch - case 。例如:

int num = 5;

if (num > 0) {

printf("Positive\n");

} else if (num < 0) {

printf("Negative\n");

} else {

printf("Zero\n");

}


switch (num) {

case 1:

printf("One\n");

break;

case 2:

printf("Two\n");

break;

default:

printf("Other\n");

}


Rust的 if - else 语句语法类似:

let num = 5;

if num > 0 {

println!("Positive");

} else if num < 0 {

println!("Negative");

} else {

println!("Zero");

}


不过Rust没有 switch - case 语句,而是使用 match 表达式,它更为强大和灵活。例如:

let num = 5;

match num {

1 => println!("One"),

2 => println!("Two"),

_ => println!("Other"),

}


match 表达式可以匹配多种类型,包括枚举类型,并且必须覆盖所有可能的情况,否则会编译错误,这有助于编写更健壮的代码。

(二)循环语句

C和C++有 for 、 while 和 do - while 循环。例如:

for (int i = 0; i < 5; i++) {

printf("%d ", i);

}


int j = 0;

while (j < 5) {

printf("%d ", j);

j++;

}


int k = 0;

do {

printf("%d ", k);

k++;

} while (k < 5);


Rust也有类似的循环结构:

for i in 0..5 {

println!("{}", i);

}


let mut j = 0;

while j < 5 {

println!("{}", j);

j += 1;

}


let mut k = 0;

loop {

println!("{}", k);

k += 1;

if k >= 5 {

break;

}

}


Rust的 for 循环使用 in 关键字来遍历范围或集合, while 循环条件判断方式与C和C++类似。Rust没有 do - while 循环,但可以使用 loop 和 break 来实现类似功能。

四、错误处理

(一)传统错误处理方式

在C和C++中,错误处理通常通过返回错误码(如C中的 errno )或抛出异常(C++)来实现。例如:

#include

#include


int divide(int a, int b) {

if (b == 0) {

throw std::runtime_error("Division by zero");

}

return a / b;

}


int main() {

try {

int result = divide(10, 0);

std::cout << "Result: " << result << std::endl;

} catch (const std::exception& e) {

std::cerr << "Error: " << e.what() << std::endl;

}

return 0;

}


这种方式存在一些问题,如异常处理可能带来性能开销,并且在C中错误码的检查容易被忽略。

(二)Rust的错误处理

Rust有一套独特的错误处理机制,主要通过 Result 枚举和 Option 枚举来实现。 Result 用于处理可能失败的操作,它有两个变体: Ok 表示成功,携带操作结果; Err 表示失败,携带错误信息。例如:

fn divide(a: i32, b: i32) -> Result {

if b == 0 {

Err(String::from("Division by zero"))

} else {

Ok(a / b)

}

}


fn main() {

match divide(10, 0) {

Ok(result) => println!("Result: {}", result),

Err(err) => println!("Error: {}", err),

}

}


Option 用于处理可能为空的值,它有 Some 和 None 两个变体。例如:

fn get_first_char(s: &str) -> Option {

if s.is_empty() {

None

} else {

Some(s.chars().next().unwrap())

}

}


let s = "hello";

match get_first_char(s) {

Some(c) => println!("First char: {}", c),

None => println!("String is empty"),

}


Rust的错误处理方式更显式,要求开发者在代码中明确处理可能的错误情况,有助于编写更可靠的代码。

五、面向对象与泛型编程

(一)面向对象特性

C++是一门支持面向对象编程的语言,通过类、继承、多态等特性实现面向对象的设计。例如:

class Animal {

public:

virtual void speak() {

std::cout << "Animal speaks" << std::endl;

}

};


class Dog : public Animal {

public:

void speak() override {

std::cout << "Dog barks" << std::endl;

}

};


Rust没有传统意义上的类和继承,但通过 trait 和结构体来实现类似的面向对象特性。 trait 定义了一组方法签名,结构体可以实现这些 trait 。例如:

trait Animal {

fn speak(&self);

}


struct Dog;


impl Animal for Dog {

fn speak(&self) {

println!("Dog barks");

}

}


这里 Animal 是一个 trait , Dog 结构体实现了 Animal trait ,通过这种方式实现了类似于C++中类和接口的功能。

(二)泛型编程

C++和Rust都支持泛型编程。C++通过模板来实现泛型,例如:

template

T add(T a, T b) {

return a + b;

}


int result = add(3, 5);


Rust使用泛型类型参数来实现泛型,例如:

fn add>(a: T, b: T) -> T {

a + b

}


let result = add(3, 5);


Rust的泛型参数需要指定其实现的 trait ,这里 T 需要实现 Add trait ,并且指定了加法操作的返回类型。

六、并发编程

(一)C和C++的并发编程

C和C++在并发编程方面主要依赖操作系统提供的线程库(如POSIX线程库、Windows线程库)或标准库中的线程支持(C++11引入了 头文件)。例如:

#include

#include


void hello() {

std::cout << "Hello from thread" << std::endl;

}


int main() {

std::thread t(hello);

t.join();

return 0;

}


并发编程在C和C++中面临着线程安全、数据竞争等问题,需要开发者手动进行同步和互斥操作。

(二)Rust的并发编程

Rust在并发编程方面提供了更安全和高效的支持。它通过 std::thread 模块创建线程,并且利用所有权和借用机制来确保线程安全。例如:

use std::thread;


fn hello() {

println!("Hello from thread");

}


fn main() {

let handle = thread::spawn(hello);

handle.join().unwrap();

}


Rust还引入了 Mutex (互斥锁)、 RwLock (读写锁)等同步原语来保证多线程环境下的数据安全。例如:

use std::sync::{Mutex, Arc};

use std::thread;


let data = Arc::new(Mutex::new(0));

let mut handles = vec![];


for _ in 0..10 {

let data = data.clone();

let handle = thread::spawn(move || {

let mut num = data.lock().unwrap();

*num += 1;

});

handles.push(handle);

}


for handle in handles {

handle.join().unwrap();

}


println!("Final value: {}", *data.lock().unwrap());


这里通过 Arc (原子引用计数)和 Mutex 来实现多线程对共享数据的安全访问,避免了数据竞争问题。

七、总结

通过与C和C++的类比,我们可以看到Rust在继承了传统系统编程语言高效、灵活的基础上,引入了许多创新的特性,如所有权系统、强大的类型系统、模式匹配、安全的并发编程等,这些特性使得Rust在内存安全、可靠性和并发性方面具有显著优势。对于精通C和C++的开发者来说,学习Rust不仅是掌握一门新语言,更是拓宽编程思维,提升编程能力的过程。在实际应用中,Rust适用于对性能和安全性要求极高的场景,如操作系统开发、网络编程、区块链技术等。随着Rust生态系统的不断完善和发展,相信它将在未来的编程领域中发挥越来越重要的作用。

相关推荐

4万多吨豪华游轮遇险 竟是因为这个原因……

(观察者网讯)4.7万吨豪华游轮搁浅,竟是因为油量太低?据观察者网此前报道,挪威游轮“维京天空”号上周六(23日)在挪威近海发生引擎故障搁浅。船上载有1300多人,其中28人受伤住院。经过数天的调...

“菜鸟黑客”必用兵器之“渗透测试篇二”

"菜鸟黑客"必用兵器之"渗透测试篇二"上篇文章主要针对伙伴们对"渗透测试"应该如何学习?"渗透测试"的基本流程?本篇文章继续上次的分享,接着介绍一下黑客们常用的渗透测试工具有哪些?以及用实验环境让大家...

科幻春晚丨《震动羽翼说“Hello”》两万年星间飞行,探测器对地球的最终告白

作者|藤井太洋译者|祝力新【编者按】2021年科幻春晚的最后一篇小说,来自大家喜爱的日本科幻作家藤井太洋。小说将视角放在一颗太空探测器上,延续了他一贯的浪漫风格。...

麦子陪你做作业(二):KEGG通路数据库的正确打开姿势

作者:麦子KEGG是通路数据库中最庞大的,涵盖基因组网络信息,主要注释基因的功能和调控关系。当我们选到了合适的候选分子,单变量研究也已做完,接着研究机制的时便可使用到它。你需要了解你的分子目前已有哪些...

知存科技王绍迪:突破存储墙瓶颈,详解存算一体架构优势

智东西(公众号:zhidxcom)编辑|韦世玮智东西6月5日消息,近日,在落幕不久的GTIC2021嵌入式AI创新峰会上,知存科技CEO王绍迪博士以《存算一体AI芯片:AIoT设备的算力新选择》...

每日新闻播报(September 14)_每日新闻播报英文

AnOscarstatuestandscoveredwithplasticduringpreparationsleadinguptothe87thAcademyAward...

香港新巴城巴开放实时到站数据 供科技界研发使用

中新网3月22日电据香港《明报》报道,香港特区政府致力推动智慧城市,鼓励公私营机构开放数据,以便科技界研发使用。香港运输署21日与新巴及城巴(两巴)公司签署谅解备忘录,两巴将于2019年第3季度,开...

5款不容错过的APP: Red Bull Alert,Flipagram,WifiMapper

本周有不少非常出色的app推出,鸵鸟电台做了一个小合集。亮相本周榜单的有WifiMapper's安卓版的app,其中包含了RedBull的一款新型闹钟,还有一款可爱的怪物主题益智游戏。一起来看看我...

Qt动画效果展示_qt显示图片

今天在这篇博文中,主要实践Qt动画,做一个实例来讲解Qt动画使用,其界面如下图所示(由于没有录制为gif动画图片,所以请各位下载查看效果):该程序使用应用程序单窗口,主窗口继承于QMainWindow...

如何从0到1设计实现一门自己的脚本语言

作者:dong...

三年级语文上册 仿写句子 需要的直接下载打印吧

描写秋天的好句好段1.秋天来了,山野变成了美丽的图画。苹果露出红红的脸庞,梨树挂起金黄的灯笼,高粱举起了燃烧的火把。大雁在天空一会儿写“人”字,一会儿写“一”字。2.花园里,菊花争奇斗艳,红的似火,粉...

C++|那些一看就很简洁、优雅、经典的小代码段

目录0等概率随机洗牌:1大小写转换2字符串复制...

二年级上册语文必考句子仿写,家长打印,孩子照着练

二年级上册语文必考句子仿写,家长打印,孩子照着练。具体如下:...

一年级语文上 句子专项练习(可打印)

...

亲自上阵!C++ 大佬深度“剧透”:C++26 将如何在代码生成上对抗 Rust?

...

取消回复欢迎 发表评论: