当前位置: 首页 > news >正文

WHAT - 高性能和内存安全的 Rust(一)

目录

  • 一、介绍
    • 1.1 示例代码
    • 1.2 关键特性
      • 内存安全
      • 零成本抽象:高效性能
        • 示例代码:使用迭代器的零成本抽象
        • 示例代码:泛型和单态化
        • 总结
      • 并发编程:防止数据竞争
        • Rust 并发编程示例
        • Rust 的所有权系统防止数据竞争
        • 总结
      • 丰富的类型系统
      • 包管理和构建系统
      • 社区和生态系统
    • 1.3 优劣势
      • 优点
      • 缺点
      • 总结
  • 二、发展历史
    • 2006-2010: 初期阶段
    • 2011-2014: 稳步发展
    • 2015: 1.0版本发布
    • 2016-2020: 成长和扩展
    • 2021至今: 持续改进和社区扩展
  • 三、前端开发者可以考虑学习
    • 为什么前端开发者应该考虑学习 Rust
    • 学习Rust的资源和建议
    • 学习Rust可能的挑战
    • 总结

一、介绍

Rust 是一种系统编程语言,以其高性能内存安全而著称。它由 Mozilla 开发,旨在在不牺牲性能的情况下提供内存安全。

Rust 适用于各种应用场景,包括系统级编程(如操作系统开发)、嵌入式编程、WebAssembly、命令行工具以及高性能Web服务器等。

1.1 示例代码

fn main() {println!("Hello, world!");
}fn add(x: i32, y: i32) -> i32 {x + y
}

这段代码展示了 Rust 的基本语法,包括函数定义和打印输出。

1.2 关键特性

以下是 Rust 的一些关键特性:

内存安全

Rust 通过所有权系统管理内存,以及编译时检查机制,避免了传统系统编程语言中的常见内存错误,如空指针引用悬挂指针缓冲区溢出

这些术语都是与内存管理相关的常见问题,尤其是在系统编程中经常遇到:

  1. 空指针引用(Null Pointer Dereference)

    • 定义:空指针是一个没有指向任何有效对象或内存地址的指针。空指针引用是指程序尝试访问通过空指针指向的内存。
    • 问题:这种操作会导致程序崩溃,通常是因为访问了非法的内存地址。
    • 示例:在C语言中,如果你尝试通过一个没有初始化的指针访问数据,可能会导致空指针引用。
    int *ptr = NULL;
    *ptr = 10; // 这将导致崩溃
    
  2. 悬挂指针(Dangling Pointer)

    • 定义:悬挂指针是指向已经被释放或不再有效的内存地址的指针。
    • 问题:使用悬挂指针可能会导致不可预知的行为,因为该内存地址可能已被系统重新分配给其他程序或用途。
    • 示例:在C语言中,如果你释放了一个指针所指向的内存,但仍然使用该指针,可能会产生悬挂指针。
    int *ptr = (int*)malloc(sizeof(int));
    free(ptr);
    *ptr = 10; // 悬挂指针,可能导致未定义行为
    
  3. 缓冲区溢出(Buffer Overflow)

    • 定义:缓冲区溢出发生在写入数据时超过了预先分配的内存缓冲区的边界。
    • 问题:这种错误可以覆盖相邻内存区域,可能导致数据损坏、程序崩溃,甚至可能被恶意利用进行攻击(例如,代码注入)。
    • 示例:在C语言中,如果你在数组边界之外写入数据,可能会导致缓冲区溢出。
    char buffer[10];
    strcpy(buffer, "This string is too long!"); // 超过缓冲区大小,导致溢出
    

Rust 通过其所有权系统和编译时检查机制,有效地防止了这些常见的内存管理问题。例如,Rust 中的所有权系统确保每块内存只有一个所有者,从而避免了悬挂指针和双重释放的问题。而借用检查器则确保在编译时捕获潜在的空指针引用和其他内存错误。Rust 还避免了缓冲区溢出,因为数组访问会在运行时进行边界检查。

零成本抽象:高效性能

Rust 的抽象不会带来运行时开销,允许开发者编写高层次代码而不会影响性能。

“零成本抽象” 是一个编程语言设计原则,指的是高层次的抽象在不引入额外运行时开销的情况下实现底层高效的代码。换句话说,程序员在使用高级语言特性时不需要担心这些特性会带来性能损失。

在 Rust 中,这一原则体现在多个方面:

  1. 内联和内联展开(Inlining and Inlining Expansion)

Rust 编译器会在编译时将小函数的实现内联展开到调用点,避免函数调用的开销。这确保了使用函数抽象不会导致额外的运行时开销。

  1. 泛型编程和单态化(Monomorphization)

Rust的泛型是通过单态化实现的,即在编译时为每个具体类型生成特定的代码。这意味着使用泛型不会引入运行时的多态开销,每个实例都是针对具体类型优化的。

  1. 枚举和模式匹配(Enums and Pattern Matching)

Rust的枚举类型和模式匹配功能在编译时进行优化,编译器生成的代码直接对应底层的高效实现,而不会引入额外的运行时检查。

  1. 迭代器和零开销抽象(Iterators and Zero-Cost Abstractions)

Rust的迭代器实现是零成本的。编译器会将迭代器链优化为高效的底层循环,这意味着在使用迭代器进行链式操作时不会有性能损失。例如,使用.map().filter()等方法不会比手写循环更慢。

示例代码:使用迭代器的零成本抽象
let numbers = vec![1, 2, 3, 4, 5];
let doubled: Vec<i32> = numbers.iter().map(|x| x * 2).collect();

上面的代码通过迭代器将每个元素翻倍并收集到一个新的向量中。Rust 编译器会将其优化为高效的底层代码,类似于手写的循环。

示例代码:泛型和单态化
fn add<T: std::ops::Add<Output = T>>(a: T, b: T) -> T {a + b
}let result = add(5, 10); // 使用泛型函数处理整数
let result_f = add(5.0, 10.0); // 使用泛型函数处理浮点数

在编译时,Rust 会为 add 函数的每个具体类型(例如 i32f64)生成特定的实现代码,从而避免了运行时的多态开销。

总结

零成本抽象的核心在于,程序员可以使用高级语言特性而不需要担心性能损失。Rust 通过其强大的编译器优化技术,实现了这一目标,使得程序员既能够编写高层次、易读的代码,又能享受到接近底层实现的性能。

并发编程:防止数据竞争

Rust 有内置的并发编程支持,利用其所有权系统来防止数据竞争问题,使编写安全的多线程代码变得更容易。

Rust 并发编程示例

下面是一个简单的 Rust 并发编程示例,展示如何使用线程和消息传递:

示例:使用线程和消息传递

use std::thread;
use std::sync::mpsc;
use std::time::Duration;fn main() {// 创建一个通道(sender和receiver)let (tx, rx) = mpsc::channel();// 启动一个新的线程thread::spawn(move || {let vals = vec![String::from("hello"),String::from("world"),String::from("from"),String::from("thread"),];for val in vals {tx.send(val).unwrap();thread::sleep(Duration::from_millis(500));}});// 主线程接收来自子线程的消息for received in rx {println!("Got: {}", received);}
}

在这个示例中:

  1. 我们创建了一个通道,用于在线程之间发送消息。
  2. 使用 thread::spawn 启动一个新的线程,该线程发送一系列字符串到主线程。
  3. 主线程通过通道接收并打印这些字符串。
Rust 的所有权系统防止数据竞争

数据竞争通常发生在多个线程同时访问和修改共享数据,而没有正确的同步机制。Rust 通过其所有权系统和类型检查机制在编译时防止这种情况。

所有权系统的基本概念

  1. 所有者(Owner):每个值在Rust中都有一个所有者。
  2. 借用(Borrowing):在同一时间,要么可以有多个不可变引用(&T),要么可以有一个可变引用(&mut T),但不能两者兼有。
  3. 生命周期(Lifetimes):编译器会检查引用的生命周期,以确保它们在使用时是有效的。

防止数据竞争的示例

use std::sync::{Arc, Mutex};
use std::thread;fn main() {let counter = Arc::new(Mutex::new(0));let mut handles = vec![];for _ in 0..10 {let counter = Arc::clone(&counter);let handle = thread::spawn(move || {let mut num = counter.lock().unwrap();*num += 1;});handles.push(handle);}for handle in handles {handle.join().unwrap();}println!("Result: {}", *counter.lock().unwrap());
}

在这个示例中:

  1. Arc(原子引用计数)Arc允许多个线程共享所有权。这里Arc用于在多个线程间共享Mutex
  2. Mutex(互斥锁)Mutex确保一次只有一个线程可以访问内部数据,从而防止数据竞争。

每个线程都会尝试获取Mutex的锁并增加计数器。因为Mutex确保一次只有一个线程能访问内部数据,所以不会出现数据竞争。

总结

Rust 通过其所有权系统借用检查器强类型系统在编译时防止数据竞争,使得编写安全的并发代码变得更加简单和可靠。

这样,程序员可以专注于实现逻辑,而不需要担心传统并发编程中常见的内存和数据一致性问题。

丰富的类型系统

Rust 拥有一个强大的类型系统,包括泛型、模式匹配和类型推断,帮助开发者捕捉更多的编译时错误。

包管理和构建系统

Rust 拥有名为 Cargo 的包管理和构建系统,使依赖管理和项目构建变得简单高效。

社区和生态系统

Rust 拥有一个活跃的社区和不断增长的生态系统,包括大量的开源库和工具。

1.3 优劣势

优点

  1. 内存安全

前面介绍过。

  1. 高性能

Rust 生成的机器代码接近于 C 和 C++ 的性能水平,适用于需要高性能的系统编程。

  1. 并发安全

前面介绍过。

  1. 零成本抽象

前面介绍过。

  1. 丰富的类型系统

前面介绍过。

  1. 活跃的社区和生态系统

前面介绍过。

缺点

  1. 学习曲线陡峭

Rust的所有权系统和借用检查器需要一些时间来理解和掌握,对于没有内存管理经验的程序员来说,学习曲线较陡。

  1. 编译时间较长

由于 Rust 的复杂类型检查和优化机制,编译时间相对会较长,可能会影响开发效率,尤其是在大型项目中。

  1. 生态系统尚未成熟

尽管 Rust 的生态系统在快速增长,但与 C 和 C++ 等老牌系统编程语言相比,某些领域(如嵌入式系统)的支持和库仍不够丰富。

  1. 语言复杂性

Rust 的语法和特性较为复杂,对于一些简单的任务可能显得过于繁琐,开发者需要在代码简洁性和安全性之间找到平衡。

  1. 难以理解的编译器错误信息

Rust 编译器的错误信息有时候可能难以理解,这可能会使调试和定位变得困难。

总结

Rust在内存安全、高性能和并发安全方面具有显著优势,使其成为系统编程、嵌入式开发、WebAssembly和高性能计算的理想选择。然而,陡峭的学习曲线和相对较长的编译时间是其主要的不足。对于追求高性能和安全性的开发者,Rust是一个非常有吸引力的选择;但对于简单任务和快速开发,可能需要权衡利弊。

二、发展历史

Rust是一种系统编程语言,旨在提供高性能和高可靠性的同时保证内存安全。其发展历史可以追溯到2006年,由Mozilla的Graydon Hoare发起,后由Mozilla赞助和社区共同开发。以下是Rust发展历史的关键节点:

2006-2010: 初期阶段

  • 2006年: Graydon Hoare开始构思Rust。
  • 2009年: Rust开始在Mozilla内部开发。
  • 2010年: Rust在GitHub上公开发布。

2011-2014: 稳步发展

  • 2011年: Rust发布0.1版本,标志着语言的早期原型。
  • 2012年: 发布0.2版本,开始引入类型系统和模式匹配等重要特性。
  • 2013年: Rust的编译器从最初的Ocaml实现转向使用LLVM作为后端,提升了性能和跨平台支持。
  • 2014年: 发布0.9版本,逐步完善标准库,开始引入更多社区贡献。

2015: 1.0版本发布

  • 2015年5月: Rust 1.0版本发布,这是Rust的第一个稳定版本,标志着语言的成熟和生产就绪状态。Rust 1.0引入了稳定的语法、库和工具链。

2016-2020: 成长和扩展

  • 2016年: 发布1.3版本,开始支持更快的编译时间和更多的标准库功能。
  • 2017年: 发布1.15版本,引入了编译器插件和宏系统,增强了代码生成能力。
  • 2018年: Rust 2018 Edition发布,这是Rust的一个重要里程碑,包含了大量的新特性和改进,如模块系统、异步编程支持和改进的错误消息。
  • 2019年: 引入了async/await语法,进一步完善了异步编程模型。
  • 2020年: 发布Rust 1.43版本,继续改进性能和稳定性。

2021至今: 持续改进和社区扩展

  • 2021年: Rust 2021 Edition发布,包含了更多的新特性和改进,如更多的类型推断、改进的编译器性能和更好的错误诊断。
  • 2022年及以后: Rust持续发布每六周一个的新版本,社区不断增长,越来越多的公司和项目采用Rust进行开发。

Rust的设计理念主要集中在安全、速度和并发三个方面。其独特的内存管理方式(所有权系统)和强类型系统使得Rust在保证内存安全的同时,能够提供与C/C++相当的性能。这些特性使得Rust在系统编程、嵌入式编程和高性能计算领域获得了广泛的认可和应用。

三、前端开发者可以考虑学习

Rust 可以是前端开发者学习的一种有价值的语言,尤其是那些希望拓展技能,了解底层系统编程或提升性能的开发者。

以下是一些具体原因和建议:

为什么前端开发者应该考虑学习 Rust

  1. 高性能和安全性: Rust 提供了高性能和内存安全性,对于需要优化前端性能或开发高效后端服务的前端开发者来说,这是一个很大的优势。

  2. WebAssembly: Rust 与 WebAssembly(Wasm)紧密集成,允许前端开发者编写高性能的 Web 应用程序。Wasm 可以显著提升 Web 应用的性能,特别是对于需要大量计算的任务。

WebAssembly是一种可以在浏览器中运行的低级字节码格式,可以被编译成 JavaScript、C++、Rust 等多种语言。WebAssembly可以在浏览器中运行,也可以在其他环境中运行,例如服务器端和移动设备。WebAssembly通常用于解决JavaScript无法处理的计算密集型任务,例如图形处理、游戏引擎、音频和视频编解码等。由于WebAssembly的执行速度比JavaScript快,因此可以提供更快的性能和更好的用户体验。WebAssembly还可以用于在浏览器中运行现有的本机代码,例如C++和Rust等语言编写的应用程序。
WebAssembly还可以与JavaScript相结合,以提供更好的性能和更丰富的功能。例如,开发人员可以将JavaScript用于Web应用程序的界面和逻辑,而将WebAssembly用于计算密集型任务。这种混合使用WebAssembly和JavaScript的方法可以提高Web应用程序的性能,并增强其功能。

  1. 全栈开发: 通过学习 Rust,前端开发者可以更容易地过渡到全栈开发,编写高效、安全的服务器端代码。

  2. 社区和生态系统: Rust 拥有活跃的社区和丰富的生态系统,提供了大量的学习资源和第三方库,如 Yew(一个Rust写的前端框架)和 Rocket(一个Rust的Web框架)。

学习Rust的资源和建议

  1. 官方文档: Rust官方文档是学习Rust的首选资源,包括The Rust Programming Language、Rust by Example等。

  2. 在线课程和教程: 多个在线平台提供Rust的课程,如Udemy、Coursera和YouTube上的教程。

  3. WebAssembly和Rust: 学习如何将Rust编译为WebAssembly,可以参考官方的Rust和WebAssembly书籍。

  4. 前端框架: 尝试使用Yew或Seed等Rust编写的前端框架,这些框架类似于React或Vue,可以帮助前端开发者更好地理解和应用Rust。

  5. 社区支持: 参与Rust社区活动,如RustConf、RustFest等,加入Rust的论坛、Discord频道和Reddit社区,与其他开发者交流和学习。

学习Rust可能的挑战

  1. 学习曲线: Rust的语法和概念(如所有权、借用检查器)可能比前端语言(如JavaScript)更复杂,需要一定的时间和实践来掌握。

  2. 生态系统相对较新: 与JavaScript相比,Rust在前端领域的生态系统还在发展中,可能需要更多的探索和适应。

总结

Rust是一个功能强大且安全的语言,对于希望提升性能、安全性和全栈开发能力的前端开发者来说,是一个值得学习的语言。通过利用Rust和WebAssembly的结合,前端开发者可以开发出更高效、更可靠的Web应用程序。尽管学习Rust可能会面临一些挑战,但通过丰富的学习资源和社区支持,这些挑战是可以克服的。

相关文章:

WHAT - 高性能和内存安全的 Rust(一)

目录 一、介绍1.1 示例代码1.2 关键特性内存安全零成本抽象&#xff1a;高效性能示例代码&#xff1a;使用迭代器的零成本抽象示例代码&#xff1a;泛型和单态化总结 并发编程&#xff1a;防止数据竞争Rust 并发编程示例Rust 的所有权系统防止数据竞争总结 丰富的类型系统包管理…...

八、C#运算符

C#运算符 晕杜甫是一种告诉编辑器执行特定的数学或逻辑操作的符号。C#有丰富的内置运算符&#xff0c;分类如下&#xff1a; 算术运算符关系运算符逻辑运算符位运算符赋值运算符其他运算符 算术运算符 下表显示了 C# 支持的所有算术运算符。假设变量 A 的值为 10&#xff0c…...

【HiveSQL】join关联on和where的区别及效率对比

测试环境&#xff1a;hive on spark spark版本&#xff1a;3.3.1 一、执行时机二、对结果集的影响三、效率对比1.内连接1&#xff09;on2&#xff09;where 2.外连接1&#xff09;on2&#xff09;where 四、总结PS 一、执行时机 sql连接中&#xff0c;where属于过滤条件&#…...

如何解决windows自动更新,释放C盘更新内存

第一步&#xff1a;首先关闭windows自动更新组件 没有更新windows需求&#xff0c;为了防止windows自动更新&#xff0c;挤占C盘空间&#xff0c;所以我们要采取停止Windows Update服务。按下WinR打开运行对话框&#xff0c;输入services.msc&#xff0c; 然后按Enter。在服务…...

初学51单片机之PWM实例呼吸灯以及遇到的问题(已解答)

PWM全名Pulse Width Modulation中文称呼脉冲宽度调制 如图 这是一个周期10ms、频率是100HZ的波形&#xff0c;但是每个周期内&#xff0c;高低电平宽度各不相同&#xff0c;这就是PWM的本质。 占空比是指高电平占整个周期的比列,上图第一个波形的占空比是40%&#xff0c;第二个…...

手机天线都去哪里了?

在手机的演变历程中&#xff0c;天线的设计和位置一直是工程师们不断探索和创新的领域。你是否好奇&#xff0c;现在的手机为什么看不到那些曾经显眼的天线了呢&#xff1f; 让我们一起揭开这个谜题。 首先&#xff0c;让我们从基础开始&#xff1a;手机是如何发出电磁波的&…...

计算机网络 —— 应用层(电子邮件)

计算机网络 —— 应用层&#xff08;电子邮件&#xff09; 电子邮件发送电子邮件的过程SMTP特性工作流程 电子邮件格式MIME关键组件工作方式 POP/IMAPPOP&#xff08;邮局协议&#xff09;IMAP&#xff08;因特网邮件访问协议&#xff09; 基于万维网的电子邮箱特点优势常见的基…...

Java18新特性(极简)

一、引言 自1995年Java语言首次亮相以来&#xff0c;它已经成为企业级应用、移动应用和游戏开发等领域不可或缺的一部分。随着技术的不断进步&#xff0c;Java也在持续演化&#xff0c;每个新版本都带来了诸多新特性和性能优化&#xff0c;旨在提升开发者的编程效率和应用程序的…...

vscode连接ssh远程服务器

当使用Visual Studio Code (VSCode) 连接SSH远程服务器时&#xff0c;可以遵循以下步骤。这些步骤将帮助你设置并连接到远程服务器&#xff0c;包括免密登录的设置&#xff08;如果需要&#xff09;。 一、安装并配置Remote-SSH插件 下载并安装VSCode&#xff1a;确保你已经下…...

【趣味测试】

编程过程中遇到的趣味知识 1 Cpp 1.1 浮点数计算 if (0.1 0.2 0.3) {std::cout << "0.1 0.2 0.3 true" << std::endl;} else {std::cout << "0.1 0.2 0.3 false" << std::endl;}if (0.1 0.3 0.4) {std::cout << &…...

数据结构经典面试之数组——C#和C++篇

文章目录 1. 数组的基本概念与功能2. C#数组创建数组访问数组元素修改数组元素数组排序 3. C数组创建数组访问数组元素修改数组元素数组排序 4. 数组的实际应用与性能优化5. C#数组示例6. C数组示例总结 数组是编程中常用的数据结构之一&#xff0c;它用于存储一系列相同类型的…...

docker的基本知识

文章目录 前言docker的基本知识1. docker 的底层逻辑2. docker 的核心要素2.1. 镜像的基本概念:2.2. 容器的基本概念:2.3. 仓库的基本概念: 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三连啊&#xff0c;写作不易啊^ _ ^。   …...

React Native性能优化红宝书

一、React Native介绍 React Native 是Facebook在React.js Conf2015 推出的开源框架&#xff0c;使用React和应用平台的原生功能来构建 Android 和 iOS 应用。通过 React Native&#xff0c;可以使用 JavaScript 来访问移动平台的 API&#xff0c;使用 React 组件来描述 UI 的…...

后端不提供文件流接口,前台js使用a标签实现当前表格数据(数组非blob数据)下载成Excel

前言&#xff1a;开发过程中遇到的一些业务场景&#xff0c;如果第三方不让使用&#xff0c;后端不提供接口&#xff0c;就只能拿到table数据(Array)&#xff0c;实现excel文件下载。 废话不多说&#xff0c;直接上代码&#xff0c;方法后续自行封装即可&#xff1a; functio…...

如何使用ChatGPT辅助设计工作

文章目录 设计师如何使用ChatGPT提升工作效率&#xff1f;25个案例告诉你&#xff01;什么是 prompt&#xff1f;咨询信息型 prompt vs 执行任务 prompt编写出色 prompt 的基本思路撰写 prompt 的案例和技巧1、将 ChatGPT 视作专业人士2、使用 ChatGPT 创建表单3、使用 ChatGPT…...

hadoop服务器启动后无法执行hdfs dfs命令

集群启动后&#xff0c;无法正常使用hdfs的任何命令。使用jps查看进程&#xff0c;发现namenode没有启动&#xff0c;然后再进入到Hadoop的相应目录&#xff0c;打开里面的logs文件 打开Hadoop的master的log 再使用vi编辑器查看&#xff08;也可以用less或者more命令查看&#…...

Flink 1.19.1 standalone 集群模式部署及配置

flink 1.19起 conf/flink-conf.yaml 更改为新的 conf/config.yaml standalone集群: dev001、dev002、dev003 config.yaml: jobmanager address 统一使用 dev001&#xff0c;bind-port 统一改成 0.0.0.0&#xff0c;taskmanager address 分别更改为dev所在host dev001 config.…...

【深度学习】GELU激活函数是什么?

torch.nn.GELU 模块在 PyTorch 中实现了高斯误差线性单元&#xff08;GELU&#xff09;激活函数。GELU 被用于许多深度学习模型中&#xff0c;包括Transformer&#xff0c;因为它相比传统的 ReLU&#xff08;整流线性单元&#xff09;函数能够更好地近似神经元的真实激活行为。…...

如何编译和运行您的第一个Java程序

​ 如何编译和运行您的第一个Java程序 让我们从一个简单的java程序开始。 简单的Java程序 这是一个非常基本的java程序&#xff0c;它会打印一条消息“这是我在java中的第一个程序”。 ​ public class FirstJavaProgram {public static void main(String[] args){System.…...

vscode用vue框架写一个登陆页面

目录 一、创建登录页面 二、构建好登陆页面的路由 三、编写登录页代码 1.添加基础结构 2.给登录页添加背景 3.解决填充不满问题 4.我们把背景的红颜色替换成背景图&#xff1a; 5.在页面中央添加一个卡片来显示登录页面 6.设置中间卡片页面的左侧 7.设置右侧的样式及…...

腾讯云API安全保障措施?有哪些调用限制?

腾讯云API的调用效率如何优化&#xff1f;怎么使用API接口发信&#xff1f; 腾讯云API作为腾讯云提供的核心服务之一&#xff0c;广泛应用于各行各业。然而&#xff0c;随着API应用的普及&#xff0c;API安全问题也日益突出。AokSend将详细探讨腾讯云API的安全保障措施&#x…...

在建设工程合同争议案件中,如何来认定“竣工验收”?

在建设工程合同争议案件中&#xff0c;如何来认定“竣工验收”&#xff1f; 建设工程的最终竣工验收&#xff0c;既涉及在建设单位组织下的五方单位验收&#xff0c;又需政府质量管理部门的监督验收以及竣工验收备案&#xff0c;工程档案还需递交工程所在地的工程档案馆归档。…...

Linux:多线程中的互斥与同步

多线程 线程互斥互斥锁互斥锁实现的原理封装原生线程库封装互斥锁 死锁避免死锁的四种方法 线程同步条件变量 线程互斥 在多线程中&#xff0c;如果存在有一个全局变量&#xff0c;那么这个全局变量会被所有执行流所共享。但是&#xff0c;资源共享就会存在一种问题&#xff1…...

数据仓库之主题域

数据仓库的主题域&#xff08;Subject Area&#xff09;是按照特定业务领域或主题对数据进行分类和组织的方式。每个主题域集中反映一个特定的业务方面&#xff0c;使得数据分析和查询更加清晰和高效。主题域通常与企业的关键业务过程相关&#xff0c;能够帮助用户在数据仓库中…...

【简易版tinySTL】 vector容器

文章目录 基本概念功能思路代码实现vector.htest.cpp 代码详解变量构造函数析构函数拷贝构造operatorpush_backoperator[]insertprintElements 本实现版本 和 C STL标准库实现版本的区别&#xff1a; 基本概念 vector数据结构和数组非常相似&#xff0c;也称为单端数组vector与…...

BRAVE:扩展视觉编码能力,推动视觉-语言模型发展

视觉-语言模型&#xff08;VLMs&#xff09;在理解和生成涉及视觉与文本的任务上取得了显著进展&#xff0c;它们在理解和生成结合视觉与文本信息的任务中扮演着重要角色。然而&#xff0c;这些模型的性能往往受限于其视觉编码器的能力。例如&#xff0c;现有的一些模型可能对某…...

使用 Verdaccio 建立私有npm库

网上有很多方法,但很多没标注nginx的版本所以踩了一些坑,下方这个文档是完善后的,对linux不是很熟练,所以不懂linux不会搭建的跟着做就可以了 搭建方法 首先需要一台云服务器 以139.196.226.123为例登录云服务器 下载node cd /usr/local/lib下载node 解压 下载 wget https://…...

个人职业规划(含前端职业+技术线路)

1. 了解自己的兴趣与长处 喜欢擅长的事 职业方向 2. 设定长期目标&#xff08;5年&#xff09; 目标内容 建立自己的品牌建立自己的社交网络 适量参加社交活动&#xff0c;认识更多志同道合的小伙伴寻求导师指导 建立自己的作品集 注意事项 每年元旦进行审视和调整永葆积极…...

LeetCode | 344.反转字符串

设置头尾两个指针&#xff0c;依靠中间变量temp交换头尾指针所指元素&#xff0c;头指针后移&#xff0c;尾指针前移&#xff0c;直到头尾指针重合或者头指针在尾指针后面一个元素 class Solution(object):def reverseString(self, s):""":type s: List[str]:r…...

一步一步用numpy实现神经网络各种层

1. 首先准备一下数据 if __name__ "__main__":data np.array([[2, 1, 0],[2, 2, 0],[5, 4, 1],[4, 5, 1],[2, 3, 0],[3, 2, 0],[6, 5, 1],[4, 1, 0],[6, 3, 1],[7, 4, 1]])x data[:, :-1]y data[:, -1]for epoch in range(1000):...2. 实现SoftmaxCrossEntropy层…...

wordpress 双栏主题/满十八岁可以申请abc认证吗

javahome路径写成死路径&#xff1b;classpath路径写死。...

网站下载到本地/开发一个小程序一般需要多少钱呢

(查询 Google 所支持的所有国家的代码,并以 zh-cn 简体中文显示)1.邮政编码法&#xff1a;&#xff08;支持美国地区&#xff09; http://www.google.com/ig/api?hlzh-cn&weather94043   (94043 为 山景城, 美国加州 的邮政编码) 2.经纬度坐标作法: http://www.google.c…...

遵义市 网站建设/电商怎么做推广

Jan. January Feb. February Mar. March Apr. April May. May Jun. June Jul. July Aug. August Sep. September Oct. October Nov. November Dec. December转载于:https://www.cnblogs.com/kakaisgood/p/11096105.html...

获取网站物理路径/培训心得体会300字

今天遇到一个很奇怪的的现象&#xff0c;在tabcontrol里切换到pagelayout时&#xff0c;程序会闪退&#xff0c;试了下之前的程序&#xff0c;没有问题&#xff0c;去网上搜了一下&#xff0c;也没人有这样的问题&#xff0c;然后就开始实验&#xff0c;添加一个控件&#xff0…...

锦州网站建设预订/网站需要怎么优化比较好

一个整数总能够拆分为2的幂的和。比如&#xff1a; 7124 71222 71114 711122 7111112 71111111 总共同拥有6种不同的拆分方式。 再比方&#xff1a;4能够拆分成&#xff1a;4 4&#xff0c;4 1 1 1 1&#xff0c;4 2 2&#xff0c;4112。用f(n)表示n的不同拆分的种…...

深圳做网站佰达科技三十/免费网站生成器

本文实例为大家分享了js实现select二级联动下拉菜单&#xff0c;供大家参考&#xff0c;具体内容如下String path request.getContextPath();String basePath request.getScheme()"://"request.getServerName()":"request.getServerPort()path"/&qu…...