rust函数指针和闭包异同探索随笔

//rust需要在编译时确定某个类型的值究竟会占据多少内存,而且同一类型的所有值都必须使用相同大小的内存,否则编译无法进行。

//对于DST动态大小类型在编译器期间无法得知其确切大小,所以直接定义此种类型的变量,rust编译无法通过!

//那么如何应对?基本上采用:引用、智能指针、impl等方式即可解决。

//rust函数是一等公民,当然可以作为参数传递存储返回。

//rust函数指针fn是rust本身具有的数据类型之一,而闭包是trait,确切说是三个trait: Fn/FnMut/FnOnce.

//因为rust函数指针impl了全部三个闭包trait, 我们总是可以把函数指针用作实参传递给一个声明接受闭包的函数。

//rust函数指针可以指向“非捕获型闭包” , 我的理解:在rust中“捕获型闭包”因为需要记录被捕获的值,所以需要创建一个结构体记录之!所以“捕获型闭包”实际上是一个结构体;

//而“非捕获型闭包”实际上就是一个函数,因为他需要的值都是通过调用传参获得,不需要记录什么,所以只需要一个函数指针记录执行入口即可。

//切记:闭包可以捕获外部变量,而函数不可以。

//当然以上是我查阅资料和代码试验后的个人理解,

//水平有限,若有谬误,希望指正。

//详细解释请仔细阅读rust编译报错提示信息即可。

//rust函数指针例子

fn add_one(x:i32) -> i32 {

    x+1

}

//case1: 把函数指针用作函数参数,传参, 编译通过。

fn do_twice(f: fn(i32)->i32, arg:i32)->i32{

    f(arg) + f(arg)

}

//case2:把函数指针用作函数返回值,编译通过。

fn take_a_function_pointer() -> fn(i32)->i32 {

    add_one

}

//rust闭包例子

//Fn/FnMut/FnOnce是闭包的三个trait.

//case3: 通过trait object返回闭包,编译通过。

fn returns_closure_by_trait_object()-> Box<dyn Fn(i32)->i32> {

    Box::new(|x| x+1)

}

//case4: 编译错误:rust编译器无法获知闭包trait编译期已知大小,请使用Box或者impl.

//fn returns_closure3()->  Fn(i32)->i32 {

//    |x| x+1

//}

//case5: 把函数指针用作返回值类型,可以返回函数或非捕获型闭包,编译通过!

fn returns_closure2_by_fn_pointer() -> fn(i32)->i32 {

    |x| x+1

}

//case6: 编译错误:rust编译器无法获知闭包trait编译期已知大小,请使用泛型或者impl或者&dyn trait object.

//fn take_closure(f: Fn(i32)->i32, arg:i32) -> i32{

//    f(arg)

//}

//case7:编译通过,通过函数指针类型形参接受非捕获型闭包或函数。

fn take_fn_or_closure_by_function_pointer(f: fn(i32)->i32, arg:i32) -> i32{

    f(arg)

}

//case8:编译通过,通过闭包trait形参接受函数指针和闭包

fn take_fn_or_closure_by_closure_trait(f: impl Fn(i32)->i32, arg:i32) -> i32 {

    f(arg)

}

fn main() {

    //for case1.

    let answer = do_twice(add_one, 5);

    println!("the function pointer answer is : {}", answer);

    //for case2.

    println!("-----------------------");

    let fp = take_a_function_pointer();

    let answer2 = fp(5);

    println!("to return function pointer answer is :{}", answer2);

     //----------------------------------

     //for case3.

    let ac = returns_closure_by_trait_object();

    let answer3 = ac(5);

    println!("to return Box dyn closure answer is: {}", answer3);

    //for case5.

    println!("-----------------------");

    let acp = returns_closure2_by_fn_pointer();

    let answer3 = acp(5);

    println!("to return closure by the function pointer answer is : {}", answer3);

    //for case7.

    println!("-----------------------");

    //函数指针可以指向“非捕获型闭包”,我理解“捕获型闭包”是通过:获取所有权(包括值复制)、可变借用、不可变借用等手段而非调用传参方式获取值。打开注释,注意编译报错信息。

    let answer4 = take_fn_or_closure_by_function_pointer(|x|x+1, 5);

    println!("to take closure by function pointer answer is : {}", answer4);

    let answer5 = take_fn_or_closure_by_function_pointer(add_one, 5);

    println!("to  take function by fn pointer answer is: {}", answer5);

    //函数指针不可以指向“捕获型闭包”,编译报错。

    //let a = 5;

    //let answer6 = take_fn_or_closure_by_function_pointer(|x| x+a, 5);

    //let a_str = String::from("hello world!");

    //let answer7 = take_fn_or_closure_by_function_pointer(|x| {println!("{}",a_str); x+1}, 5);


 

    //for case8.

    println!("-----------------------");

    let answer8 = take_fn_or_closure_by_closure_trait(|x|x+1, 5);

    println!("to take a closure by the closure trait, answer is: {}", answer8);

    let answer9 = take_fn_or_closure_by_closure_trait(add_one, 5);

    println!("to take a function by the closure trait, answer is: {}", answer9);



 

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/32573.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

qml:导入B站Up主的FluentUI插件

文章目录 文章介绍如何加载1、下载代码2、官方文档和组件介绍 运行FluentUI新建自己的qml项目&#xff0c;并导入FluentUI调用组件&#xff0c;展示效果图 文章介绍 up主“会磨刀的小猪”模仿微软Fluent风格写的界面&#xff0c;可以理解为用qt和qml写出的win10/win11风格的界…

Ubuntu 之Glade图形化设计器

演示环境说明&#xff1a;本机使用Windows 11 家庭版本搭载 Ubuntu 22.04.4 LTS 子系统&#xff0c;同时并安装Ubuntu桌面虚拟化软件XLaunch。 如果没有搭建好上述问题&#xff0c;请参考&#xff1a;windows11子系统Ubuntu 22.04.4子安装图形化界面 Glade是什么&#xff1f;…

im即时通讯软件系统,私有化部署国产化信创适配安全可控

私有化部署IM即时通讯软件系统是许多企业为了确保数据安全、控制隐私保护、提升灵活性而考虑的重要选择之一。信创适配安全可控是企业在私有化部署IM即时通讯软件系统时需要关注的关键点。本文将探讨私有化部署IM即时通讯软件系统的意义、信创适配的重要性&#xff0c;以及如何…

使用Vercel 搭建自己的Dashy导航页

背景 Dashy 是一个开源的自托管导航页面配置服务&#xff0c;它具有易于使用的可视化编辑器、状态检查、小工具和主题等功能。用户可以利用 Dashy 将自己常用的一些网站聚合起来&#xff0c;形成一个个性化的导航页面。 同类的竞品还有Heimdall, Flare 等。 可以通过Docker 等…

OneNote 作为恶意软件分发新渠道持续增长

目前&#xff0c;Office 文件已经默认禁用宏代码&#xff0c;攻击者开始转向利用其他微软的软件产品来进行恶意 Payload 投递。默认情况下&#xff0c;OneNote 应用也包含在 Office 2019 和 Microsoft 365 软件中&#xff0c;所以 OneNote 文件越来越受到攻击者的青睐。如果有人…

(南京观海微电子)——TFT LCD压合技术

TFT-LCD TFT-LCD open cell后段制程主要指的是将驱动IC和PCB压合至液晶板上&#xff0c;这个制程主要由三个步骤组成&#xff1a; 1.ACF (Anisotropic Conductive Film)的涂布。 在液晶板需要压合驱动IC的地方涂布ACF&#xff0c;ACF又称异方性导电胶膜&#xff0c;特点是上下…

vue3 Elementplus 动态路由菜单不跳转问题

问题描述 vue3 Elementplus 使用component: () > import(/views/${item.componentPath}.vue)加载动态路由菜单不跳转, 报错 Unknown variable dynamic import: …/views/system/user/index.vue at dynamic-import-helper.js:7:96 解决 使用 使用 let modules impor…

IntelliJ IDEA 中显示或隐藏类中的方法

直接上图 左侧找到Project->右键->勾选Show Members即可显示 没有勾选的状态 勾选后的状态

模板初阶【C++】

文章目录 模板的作用模板的原理模板分为两大类——函数模板和类模板函数模板语法函数模板实例化模板函数的方式模板函数的类型转换既有函数模板又有已经实现的函数&#xff0c;会优先调用哪一个&#xff1f; 类模板语法模板类实例化对象模板类的模板参数可以有缺省值类模板中的…

如何获取特定 HIVE 库的元数据信息如其所有分区表和所有分区

如何获取特定 HIVE 库的元数据信息如其所有分区表和所有分区 1. 问题背景 有时我们需要获取特定 HIVE 库下所有分区表&#xff0c;或者所有分区表的所有分区&#xff0c;以便执行进一步的操作&#xff0c;比如通过 使用 HIVE 命令 MSCK REPAIR TABLE table_name sync partiti…

【计算机毕业设计】194高校学习助手微信小程序

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

Node.js实现短链接(ShortLink):shortid、epxress让URL更简单

文章目录 一、短链接介绍二、插件介绍1、epxress2、shortid 三、实现方案1、安装依赖&#xff1a;2、实现原理 四、示例代码五、测试生产短链接 一、短链接介绍 短链接是指仅包含一个网址的链接形式&#xff0c;通俗一些就是将一个很长很复杂的的网址变成一个简短易记的链接。…

Android Gradle开发与应用 (一) : Gradle基础

目录 一、什么是Gradle? 1.1 Gradle简介 1.2 Gradle的特点 二、Gradle的安装与配置 2.1 安装Gradle 2.1.1 下载Gradle 2.1.2 解压安装包 2.1.3 配置环境变量 2.1.4 验证安装 2.2 配置Gradle 2.2.1 配置Gradle版本 2.2.2 配置Gradle脚本 三、Gradle的构建文件结构…

TCP数据包长度范围介绍

在TCP协议中&#xff0c;数据被分割成称为TCP数据包&#xff08;TCP segment&#xff09;的单元进行传输。 TCP数据包的长度没有固定的范围&#xff0c;它可以根据网络和应用程序的需求而变化。 一般来说&#xff0c;TCP数据包的长度受到以下几个因素的影响&#xff1a; 最大传…

有趣的 Oracle JDBC 驱动包命名问题 - ojdbc6 和 ojdbc14 哪个新?!

有趣的 Oracle JDBC 驱动包命名问题 - ojdbc6 和 ojdbc14 哪个新?! 1 背景概述 最近协助一个小兄弟排查了某作业使用 sqoop 采集 oracle 数据的失败问题&#xff0c;问题现象&#xff0c;问题原因和解决方法都挺直观&#xff0c;但在此过程中发现了一个有趣的 Oracle JDBC 驱…

第三十六章 添加和使用自定义标题元素 - 指定 XData 块中支持的标头元素

文章目录 第三十六章 添加和使用自定义标题元素 - 指定 XData 块中支持的标头元素指定 XData 块中支持的标头元素Details 第三十六章 添加和使用自定义标题元素 - 指定 XData 块中支持的标头元素 指定 XData 块中支持的标头元素 如果使用 SOAP 向导根据给定的 WSDL 创建 Web …

微软Edge浏览器:全面解析与深度探索

一、引言 随着互联网的飞速发展&#xff0c;浏览器作为我们访问网络世界的重要工具&#xff0c;其性能和功能越来越受到用户的关注。微软Edge浏览器自推出以来&#xff0c;凭借其高效的性能、丰富的功能和安全的浏览环境&#xff0c;赢得了众多用户的青睐。本文将全面解析微软…

面试专区|【47道ES67高频题整理(附答案背诵版)】

1.ES5、ES6&#xff08;ES2015&#xff09;有什么区别? ES5&#xff08;ECMAScript 5&#xff09;和ES6&#xff08;也称为ECMAScript 2015&#xff09;是JavaScript语言的两个版本&#xff0c;它们之间有一些重要的区别和改进&#xff1a; let 和 const 关键字&#xff1a; …

Kylin系列:架构和高级功能详解

目录 一、Kylin的架构 1.1 总体架构概述 1.2 数据源 1.3 元数据存储 1.4 构建引擎 1.5 存储引擎 1.6 查询引擎 1.7 用户接口 二、Kylin的高级功能 2.1 多维立方体(Cube) 2.1.1 Cube的定义 2.1.2 Cube的构建 2.2 查询优化 2.3 数据模型和星型模式 2.3.1 数据模…

使用SPI驱动数码管

代码&#xff1a; 7-seg.c /*《AVR专题精选》随书例程3.通信接口使用技巧项目&#xff1a;改进的延时法实现半双工软件串口文件&#xff1a;7seg.c说明&#xff1a;SPI控制数码管驱动文件作者&#xff1a;邵子扬时间&#xff1a;2012年12月15日*/#include <avr/io.h>ex…