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

std::function的简易实现

本节我们来实现一个简易的std::function

我们知道std::function是用来包装可调用对象的,在C++中,可调用对象包括 普通函数lambda表达式重载了()操作符的类对象类静态函数类成员函数这几类。

C++程序的编译顺序:预处理(xxx.i) 编译(xxx.s) 汇编(xxx.obj) 链接(xxx.exe / linux可执行文件)
模板类(函数)的实例化,是在 预处理之后,编译之前;所以模板并不是真正可执行的C++代码,应该将它理解成一个模具,我们使用什么样的材料(模板参数),就会生成什么样的代码;

模板类(函数)的模板参数,其实只是一个类型占位符;我们可以赋予这个占位符任何意义,模板类(函数)的实现也是基于这个我们赋予的意义来进行编写的;所以在使用模板类(函数)时,只要传入的模板参数能满足我们赋予的意义,这个模板类(函数)就能正常运行

模板函数的模板参数是可以使用编译器自动推导,而模板类的模板参数,却需要显示指定,无法使用自动推导


首先回顾一下 std::function的使用,举个比较简单的例子:

	int func(int x, int y) { return x + y; }std::function<int, (int, int)> f1(func);		//使用1std::function<int, (int, int)> f2([int x, int y](){ return x + y; });	//使用2class Test{int memFunc(int x, int y) { return x + y; }}Test t;std::function<int, (int, int)> f2(&Test::memFunc, &t);		//使用3

可以看到,我们给std::function传递模板参数时,模板参数的形式为 ReturnType (Params)
观察使用1 使用2可以发现,构造函数的参数形式为一个可以被执行的函数(1中为普通函数,2中为lambda表达式)

结合我们以上发现的特点,最容易想到形式是:
模板参数传递 返回值类型 参数包, 构造函数传递一个可执行的函数; 然后重载 ()操作符,将参数包转发到构造函数传递的可调用函数中去,执行一下该函数即可;

//代码段1
template <typename Ret, typename... Args>
struct myfunction
{template <typename Functor>myfunction(Functor&& func){// 将 func 存储到 m_func 中, // m_func(std::forward<Functor>(func));}Ret operator()(Args&&... args){//调用 m_func 即可
//        return m_func(std::forward<Args>(args)...);}
};

这里其实存在一个问题,那就是我们其实并没有这个m_func成员变量;这个成员变量的类型应该是和构造函数的模板参数一个类型(Functor类型),但是这个Functor只对构造函数可见,所以我们没办法定义一个 Functor m_func;的成员变量;

那怎么解决?

自然而然就会想到,将 Functor模板参数从构造函数的模板参数,移动到类模板参数中去,扩大了作用域;自然构造函数就不再是模板函数

//代码段2
template <typename Functor, typename Ret, typename... Args>
struct myfunction
{myfunction(Functor&& func){m_func(std::forward<Functor>(func));}Ret operator()(Args&&... args){return m_func(std::forward<Args>(args)...);}Functor m_func;
};

这样做固然可以,但是使用时我们是不是得传递这个模板类的模板参数,还是使用以上的例子,我们现在必须得这样使用这个模板类 myfunction<decltype(func), int (int,int)> f;, 其中第一个模板参数Functor, 需要我们自己显示指定;这显然会加大使用负担,且与std::function<int (int, int)>的形式不符;

结合以上的思考,将第一个模板参数Functor放到构造函数中去,由编译器进行推导,而不是定义myfunction时显示指定类型,这是比较合理的,也更符合使用习惯;所以我们还是要在代码段1的基础上进行改进;
现在问题变成了: 怎样才能让代码段1支持存储Functor类型的成员变量呢?

有了以上的思考,我们知道,将模板参数的作用域从函数提升到中,就可以扩大可见性
我们可以构造一个辅助类,将构造函数的模板参数变成辅助类的类模板参数,这样就可以将辅助类的类模板参数存储在辅助类中,然后想办法在myfunction中存储辅助类的指针或对象,在myfunction重载的()中调用存储在辅助类中的可调用函数,代码如下:

template <typename Ret, typename... Args>
struct myfunction
{template <typename Functor>myfunction(Functor&& func){m_pCallAble = new CallAble<Functor>(std::forward<Functor>(func));}Ret operator()(Args&&... args){if (m_pCallAble != nullptr){return m_pCallAble->invoke(std::forward<Args>(args)...);}}//辅助类基类struct ICallAble{virtual ~ICallAble() = default;virtual Ret invoke(Args&&... args) = 0;};//辅助类, 用于存储myfunction构造函数中的 可执行函数template <typename Functor>struct CallAble{CallAble(Functor&& func) : m_func(func) {}Ret invoke(Args&&... args) override{return m_func(std::forward<Args>(args)...);}Functor m_func;};ICallAble* m_pCallAble = nullptr;
};

OK,以上代码我们成功将myfunction构造函数的参数存储到了 m_pCallAble 变量中,到这里为止,我们这个myfunction类已经可以包装 普通函数 lambda表达式 重载了()的类对象 类静态函数
但是还无法包装类成员函数,我们观察最初的例子中的 形式3,不难发现这种形式的构造函数需要传递一个可调用函数,以及一个类对象, 所以我们还需要给myfunction加一个重载的构造函数

    template<typename Functor, typename Class>myfunction(Functor&& func, Class* obj){}

当然我们可以继续使用 m_pCallAble成员变量,此时 new的就不再是 CallAble类对象了,因为这个类只能存储Functor模板参数,无法存储Class模板参数;所以我们继续派生自 ICallAble,写一个可以存储这两个参数的模板类

	//重载的构造函数template<typename Functor, typename Class>myfunction(Functor&& func, Class* obj){m_pCallAble = new MemCallAble<Functor, Class>(std::forward<Functor>(func), obj);}//包装类成员函数的派生类template <typename Functor, typename Class>struct MemCallAble : public ICallAble{MemCallAble(Functor&& func, Class* obj): m_func(std::forward<Functor>(func)), m_obj(obj){}Ret invoke(Args&&... args) override{return (m_obj->*m_func)(std::forward<Args>(args)...);}Functor m_func;Class* m_obj = nullptr;};

这样我们的myfunction就可以类似形式3,来包装类成员函数了,到此实现了基本功能, 贴出以上思考后的代码:

template <typename T>
class myfunction;template <typename Ret, typename... Args>
class myfunction<Ret(Args...)>
{
public:myfunction() {}template <typename Function>myfunction(Function&& func) : m_ptr { new CallAble<Function>(std::forward<Function>(func)) } {}template <typename Function, typename Class>myfunction(Function&& func, Class* obj) : m_ptr { new MemCallAble<Function, Class>(std::forward<Function>(func), obj) } {}myfunction(const myfunction& rhs) { m_ptr = rhs.m_ptr->clone(); }myfunction& operator=(const myfunction& rhs){if (this == &rhs){return *this;}delete m_ptr;m_ptr = rhs.m_ptr ? rhs.m_ptr->clone() : nullptr;return *this;}~myfunction(){if (m_ptr){delete m_ptr;}}myfunction(myfunction&& rhs){if (m_ptr ){delete m_ptr;}m_ptr = rhs.m_ptr;rhs.m_ptr = nullptr;}Ret operator()(Args... args) { return m_ptr->invoke(std::forward<Args>(args)...); }private://辅助类基类,使用继承方式, 擦除掉构造函数的模板参数;这样可以在此类中定义 ICallAble 指针struct ICallAble{virtual Ret invoke(Args&&... args) = 0;virtual ICallAble* clone() = 0;virtual ~ICallAble() = default;};//包装一般可调用对象, 普通函数、lambda表达式、类静态函数、重载了()的类对象template <typename Function>struct CallAble : public ICallAble{CallAble(Function&& func) : m_real_func(std::forward<Function>(func)) {}ICallAble* clone() override { return new CallAble<Function>(std::forward<Function>(m_real_func)); }Ret invoke(Args&&... args) override{return m_real_func(std::forward<Args>(args)...);}~CallAble() = default;Function m_real_func = nullptr;    //真正执行的函数};//包装类成员函数,类成员函数需要类对象才能进行调用,所以比普通版本多一个Class类型模板参数template<typename Function, typename Class>struct MemCallAble : public ICallAble{MemCallAble(Function&& func, Class* obj) : m_real_func(std::forward<Function>(func)), m_obj(obj){}ICallAble* clone() override { return new MemCallAble<Function, Class>(std::forward<Function>(m_real_func), m_obj); }Ret invoke(Args&&... args) override{return (m_obj->*m_real_func)(std::forward<Args>(args)...);}~MemCallAble() = default;Function m_real_func = nullptr; //真正执行的函数Class* m_obj = nullptr;         //类对象};private:ICallAble* m_ptr = nullptr;
};

相关文章:

std::function的简易实现

本节我们来实现一个简易的std::function 我们知道std::function是用来包装可调用对象的&#xff0c;在C中&#xff0c;可调用对象包括 普通函数、lambda表达式、重载了()操作符的类对象、类静态函数、类成员函数这几类。 C程序的编译顺序&#xff1a;预处理(xxx.i) 编译(xxx.…...

笔试-二维数组1

应用 快递业务有N个站点&#xff0c;1<N<10000&#xff1b;站点0、站点1可达&#xff0c;记作0-1&#xff1b;如果0-1、1-2&#xff0c;则站点0、站点2可达&#xff0c;记作0-2&#xff1b;s[i][j]1表示i-j可达&#xff0c;反之s[i][j]0表示i-j不可达&#xff1b;s[i][j…...

【Pytest】生成html报告中,中文乱码问题解决方案

import pytestif __name__ "__main__":# 只运行 tests 目录下的测试用例&#xff0c;并生成 HTML 报告pytest.main([-v, -s, --htmlreport.html, tests])可以以上方式生成&#xff0c;也可以在pytest.ini中设置 [pytest] addopts --htmlreport.html --self-contai…...

汽车网络信息安全-ISO/SAE 21434解析(下)

目录 第十二~十四章 - 后开发阶段 1. 十二章节 - 生产 2. 十三章节 - 运营与维护 网络安全事件响应 更新 3. 十四章节 - 结束网络安全支持和停用 结束网络安全支持 报废 第十五章 - TARA分析方法 1. 概述 2. 资产识别 3. 威胁场景识别 4. 影响评级 5. 攻击路径分…...

局域网中 Windows 与 Mac 互相远程连接的最佳方案

由于工作需要&#xff0c;经常需要远程连接或登录到几台不同的工作用机上进行操作。 下面基于免费、高体验等基本诉求&#xff0c;简要记录几种不同场景下的实践方案选择&#xff0c;仅供参考。如您有更简单且更优的方案&#xff0c;欢迎一起探讨。 1 远程桌面连接的几种不同…...

Qt调用ffmpeg库录屏并进行UDP组播推流

基于以下参考链接&#xff0c;采用其界面和程序框架&#xff0c;实现实时推送UDP组播视频流&#xff0c;替换原拉流功能 https://blog.csdn.net/u012532263/article/details/102736700 源码在windows&#xff08;qt-opensource-windows-x86-5.12.9.exe&#xff09;、ubuntu20.…...

机器学习 - 初学者需要弄懂的一些线性代数的概念

一、单位矩阵 在数学中&#xff0c;单位矩阵是一个方阵&#xff0c;其主对角线上的元素全为1&#xff0c;其余元素全为0。单位矩阵在矩阵乘法中起到类似于数字1在数值乘法中的作用&#xff0c;即任何矩阵与单位矩阵相乘&#xff0c;结果仍为原矩阵本身。 单位矩阵的定义&…...

vscode无法格式化go代码的问题

CTRLshiftp 点击Go:Install/Update Tools 点击全选&#xff0c;OK&#xff01;...

实验二 数据库的附加/分离、导入/导出与备份/还原

实验二 数据库的附加/分离、导入/导出与备份/还原 一、实验目的 1、理解备份的基本概念&#xff0c;掌握各种备份数据库的方法。 2、掌握如何从备份中还原数据库。 3、掌握数据库中各种数据的导入/导出。 4、掌握数据库的附加与分离&#xff0c;理解数据库的附加与分离的作用。…...

基于Django的个人博客系统的设计与实现

【Django】基于Django的个人博客系统的设计与实现&#xff08;完整系统源码开发笔记详细部署教程&#xff09;✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 系统采用Python作为主要开发语言&#xff0c;结合Django框架构建后端逻辑&#xff0c;并运用J…...

PostgreSQL TRUNCATE TABLE 操作详解

PostgreSQL TRUNCATE TABLE 操作详解 引言 在数据库管理中,经常需要对表进行操作以保持数据的有效性和一致性。TRUNCATE TABLE 是 PostgreSQL 中一种高效删除表内所有记录的方法。本文将详细探讨 PostgreSQL 中 TRUNCATE TABLE 的使用方法、性能优势以及注意事项。 什么是 …...

黑盒/白盒运维监控

运维监控分为黑盒和白盒 黑盒&#xff1a;不深入代码&#xff0c;在系统角度看TPS&#xff0c;延迟等指标 白盒&#xff1a;深入代码分析&#xff0c;通过日志捕捉&#xff0c;以及主动上报告警等来进行监控 黑盒监控&#xff1a; 1. 页面功能&#xff1a;域名是否可访问&…...

日志收集Day007

1.配置ES集群TLS认证: (1)elk101节点生成证书文件 cd /usr/share/elasticsearch ./bin/elasticsearch-certutil cert -out config/elastic-certificates.p12 -pass "" --days 3650 (2)elk101节点为证书文件修改属主和属组 chown elasticsearch:elasticsearch con…...

微信小程序1.1 微信小程序介绍

1.1 微信小程序介绍 内容提要 1.1 什么是微信小程序 1.2 微信小程序的功能 1.3 微信小程序使用场景 1.4 微信小程序能取代App吗 1.5 微信小程序的发展历程 1.6微信小程序带来的机会...

Leetcode 3434. Maximum Frequency After Subarray Operation

Leetcode 3434. Maximum Frequency After Subarray Operation 1. 解题思路2. 代码实现 题目链接&#xff1a;3434. Maximum Frequency After Subarray Operation 1. 解题思路 这一题的话我们只需要考察所有的数 i i i转换为 k k k时所能够形成的最大的值。 而对于这个问题&…...

6. 使用springboot做一个音乐播放器软件项目【1.0版项目完结】附带源码~

#万物OOP 注意&#xff1a; 本项目只实现播放音乐和后台管理系统。 不分享任何音乐歌曲资源。 上一篇文章我们 做了音乐播放器后台的功能。参考地址&#xff1a; https://jsonll.blog.csdn.net/article/details/145214363 这个项目已经好几天也没更新了&#xff0c;因为临近放…...

php twig模板引擎详细使用教程

php twig模板引擎 1. 什么是Twig模板引擎 Twig是一个强大且灵活的PHP模板引擎&#xff0c;它提供了一种更简洁和可扩展的方法来创建PHP应用程序的视图层。Twig模板引擎旨在将设计与业务逻辑分离&#xff0c;并为开发人员提供一种更加清晰和易于维护的方式来构建网页。Twig由S…...

【Java设计模式-7】责任链模式:我是流水线的一员

一、责任链&#xff08;Chain of Responsibility Patten&#xff09;模式是个啥&#xff1f; 想象一下&#xff0c;你要请假。你先把请假申请交给了小组长&#xff0c;小组长一看&#xff0c;这事儿他能决定&#xff0c;就直接批了。要是小组长觉得这事儿得往上汇报&#xff0…...

Spring Boot应用中实现基于JWT的登录拦截器,以保证未登录用户无法访问指定的页面

目录 一、配置拦截器进行登录校验 1. 在config层设置拦截器 2. 实现LoginInterceptor拦截器 3. 创建JWT工具类 4. 在登录时创建JWT并存入Cookie 二、配置JWT依赖和环境 1. 添加JWT依赖 2. 配置JWT环境 本篇博客将为大家介绍了如何在Spring Boot应用中实现基于JWT的登录…...

【2025年数学建模美赛F题】(顶刊论文绘图)模型代码+论文

全球网络犯罪与网络安全政策的多维度分析及效能评估 摘要1 Introduction1.1 Problem Background1.2Restatement of the Problem1.3 Literature Review1.4 Our Work 2 Assumptions and Justifications数据完整性与可靠性假设&#xff1a;法律政策独立性假设&#xff1a;人口统计…...

计算机网络之链路层

本文章目录结构出自于《王道计算机考研 计算机网络_哔哩哔哩_bilibili》 02 数据链路层 在网上看到其他人做了详细的笔记&#xff0c;就不再多余写了&#xff0c;直接参考着学习吧。 1 详解数据链路层-数据链路层的功能【王道计算机网络笔记】_wx63088f6683f8f的技术博客_51C…...

随笔十七、eth0单网卡绑定双ip的问题

在调试语音对讲过程中遇到过一个“奇怪”问题&#xff1a;泰山派作为一端&#xff0c;可以收到对方发来的语音&#xff0c;而对方不能收到泰山派发出的语音。 用wireshark抓包UDP发现&#xff0c;泰山派发送的地址是192.168.1.30&#xff0c;而给泰山派实际设置的静态地址是19…...

coffee销售数据集分析:基于时间趋势分析的实操练习

**文章说明&#xff1a;**对coffee销售数据集的简单分析练习&#xff08;时间趋势分析练习&#xff09;&#xff0c;主要是为了强化利用python进行数据分析的实操能力。属于个人的练习文章。 **注&#xff1a;**这是我第一次使用md格式编辑博客文章&#xff0c;排版上还是不是很…...

在 Vue 3 中,怎么管理环境变量

在 Vue 3 中&#xff0c;环境变量管理是通过 .env 文件来进行的&#xff0c;利用这些文件可以让开发者根据不同的环境&#xff08;开发、生产、测试等&#xff09;配置不同的变量。这一机制由 Vite 构建工具支持&#xff0c;它帮助开发者根据不同的环境需求做出相应配置。 1. …...

shallowRef和shallowReactive的用法以及使用场景和ref和reactive的区别

Vue3 浅层响应式 API 1. ref vs shallowRef 1.1 基本概念 ref: 深层响应式&#xff0c;会递归地将对象的所有属性转换为响应式shallowRef: 浅层响应式&#xff0c;只有 .value 的改变会触发更新&#xff0c;不会递归转换对象的属性 1.2 使用对比 // ref 示例 const deepRe…...

mantisbt添加修改用户密码

文章目录 问题当前版本安装流程创建用户修改密码老的方式探索阶段 问题 不太好改密码啊。貌似必须要域名要发邮件。公司太穷&#xff0c;看不见的东西不关心&#xff0c;只能改源码了。 当前版本 当前mantisbt版本 2.27 php版本 7.4.3 安装流程 &#xff08;下面流程不是…...

mysql 学习6 DQL语句,对数据库中的表进行 查询 操作

前期准备数据 重新create 一张表 create table emp(id int comment 编号,workno varchar(10) comment 工号,name varchar(10) comment 姓名,gender char comment 性别,ager tinyint unsigned comment 年龄,idcard char(18) comment 身份证号,workaddress varchar(10) c…...

零售业革命:改变行业的顶级物联网用例

mpro5 产品负责人Ruby Whipp表示&#xff0c;技术进步持续重塑零售业&#xff0c;其中物联网&#xff08;IoT&#xff09;正引领这一变革潮流。 研究表明&#xff0c;零售商们正在采用物联网解决方案&#xff0c;以提升运营效率并改善顾客体验。这些技术能够监控运营的各个方面…...

云计算的概念与特点:开启数字化时代的新篇章

在当今数字化时代,云计算(Cloud Computing)已经成为推动技术创新和业务转型的核心力量。无论是大型企业、中小型企业,还是个人用户,云计算都为其提供了高效、灵活和经济的解决方案。本文将深入探讨云计算的概念及其核心特点,帮助读者全面了解这一革命性技术。 © ivw…...

第二十一周:Mask R-CNN

Mask R-CNN 摘要Abstract文章信息研究动机Mask RCNNRoIPool与RoIAlign 双线性插值Mask Branch(FCN)其他细节Mask RCNN损失Mask分支预测 网络搭建创新点与不足总结 摘要 本篇博客介绍了Mask R-CNN&#xff0c;这是一种用于实例分割的模型&#xff0c;能够在目标检测的基础上实现…...

特朗普政府将开展新网络攻击

近日&#xff0c;特朗普政府已表态&#xff1a;减少物理战争&#xff0c;网络战将代替&#xff0c;以实现美国的全球优势。 特朗普也指示美国网络司令部可以在没有总统批准的情况下开展更广泛行动&#xff0c;尤其是应对一些突发事件&#xff0c;这其实成为了后续美国通过网络…...

Android Studio:视图绑定的岁月变迁(2/100)

一、博文导读 本文是基于Android Studio真实项目&#xff0c;通过解析源码了解真实应用场景&#xff0c;写文的视角和读者是同步的&#xff0c;想到看到写到&#xff0c;没有上帝视角。 前期回顾&#xff0c;本文是第二期。 private Unbinder mUnbinder; 只是声明了一个 接口…...

【已解决】黑马点评项目Redis版本替换过程的数据迁移

黑马点评项目Redis版本替换过程的数据迁移 【哭哭哭】附近商户中需要用到的GEO功能只在Redis 6.2以上版本生效 如果用的是老版本&#xff0c;美食/KTV的主页能正常返回&#xff0c;但无法显示内容 上次好不容易升到了5.0以上版本&#xff0c;现在又用不了了 Redis 6.2的windo…...

Maven运行任何命令都报错“Internal error: java.lang.ArrayIndexOutOfBoundsException”

今天遇到一个奇怪的问题&#xff0c;在maven工程下运行任何mvn命令都报“Internal error: java.lang.ArrayIndexOutOfBoundsException”错误&#xff0c;具体错误如下&#xff1a; $ mvn install [INFO] Scanning for projects... [ERROR] Internal error: java.lang.ArrayInd…...

电商平台爬虫开发技术分享:多年的实战经验总结

在当今数字化时代&#xff0c;电商平台的数据蕴含着巨大的商业价值。作为一名从事电商平台爬虫开发的工程师&#xff0c;我深知数据抓取的重要性及其技术挑战。经过多年的实践&#xff0c;我积累了一些宝贵的经验&#xff0c;愿意在这里与大家分享&#xff0c;希望能为同行们提…...

大模型训练策略与架构优化实践指南

标题&#xff1a;大模型训练策略与架构优化实践指南 文章信息摘要&#xff1a; 该分析全面探讨了大语言模型训练、架构选择、部署维护等关键环节的优化策略。在训练方面&#xff0c;强调了pre-training、mid-training和post-training的不同定位与目标&#xff1b;在架构选择上…...

DeepSeek-R1 蒸馏模型及如何用 Ollama 在本地运行DeepSeek-R1

在人工智能飞速发展的领域中&#xff0c;大型语言模型&#xff08;LLMs&#xff09;的出现可谓是一项重大变革。在这些模型里&#xff0c;DeepSeek - R1 及其蒸馏模型备受瞩目&#xff0c;它们融合了独特的能力与高可用性。今天我们一起聊一下 DeepSeek - R1 蒸馏模型究竟是什么…...

机器学习 ---逻辑回归

逻辑回归是属于机器学习里面的监督学习&#xff0c;它是以回归的思想来解决分类问题的一种非常经典的二分类分类器。由于其训练后的参数有较强的可解释性&#xff0c;在诸多领域中&#xff0c;逻辑回归通常用作 baseline 模型&#xff0c;以方便后期更好的挖掘业务相关信息或提…...

虚幻浏览器插件 UE与JS通信

温馨提示&#xff1a;本节内容需要结合插件Content下的2_Communication和Resources下的sample.html 一起阅读。 1. UE调用JS 1.1 JS脚本实现 该部分共两步: 导入jstote.js脚本实现响应函数并保存到 ue.interface 中 jsfunc 通过json对象传递参数&#xff0c;仅支持函数名小…...

KVM/ARM——基于ARM虚拟化扩展的VMM

1. 前言 ARM架构为了支持虚拟化做了些扩展&#xff0c;称为虚拟化扩展(Virtualization Extensions)。原先为VT-x创建的KVM(Linux-based Kernel Virtual Machine)适配了ARM体系结构&#xff0c;引入了KVM/ARM (the Linux ARM hypervisor)。KVM/ARM没有在hypervisor中引入复杂的…...

池化层Pooling Layer

1. 定义 池化是对特征图进行的一种压缩操作&#xff0c;通过在一个小的局部区域内进行汇总统计&#xff0c;用一个值来代表这个区域的特征信息&#xff0c;常用于卷积神经网络&#xff08;CNN&#xff09;中。 2. 作用 提取代表性信息的同时降低特征维度&#xff0c;具有平移…...

为AI聊天工具添加一个知识系统 之63 详细设计 之4:AI操作系统 之2 智能合约

本文要点 要点 AI操作系统处理的是 疑问&#xff08;信念问题&#xff09;、缺省&#xff08;逻辑问题&#xff09;和异常(不可控因素 ) 而 内核 的三大功能 &#xff08;资源分配/进程管理/任务调度&#xff09;以及外围的三类接口&#xff08; CLI、GUI和表面模型的 运行时…...

代码随想录——二叉树(二)

文章目录 前言二叉树最大深度二叉树的最小深度翻转二叉树对称二叉树完全二叉树的节点个数平衡二叉树二叉树的所有路径左叶子之和找左下角的值路径总和从中序与后序序列构造二叉树最大二叉树合并二叉树二叉搜索树中的搜索验证二叉搜索树二叉搜索树的最小绝对差二叉树中的众数二叉…...

一个基于Python+Appium的手机自动化项目~~

本项目通过PythonAppium实现了抖音手机店铺的自动化询价&#xff0c;可以直接输出excel&#xff0c;并带有详细的LOG输出。 1.excel输出效果: 2. LOG效果: 具体文件内容见GitCode&#xff1a; 项目首页 - douyingoods:一个基于Pythonappium的手机自动化项目&#xff0c;实现了…...

深入剖析SpringBoot启动机制:run()方法详尽解读

摘要 本文深入解析SpringBoot的启动机制&#xff0c;以run()方法为核心&#xff0c;逐步追踪并详细解释其关键步骤。首先探讨run()方法的工作原理&#xff0c;然后深入代码层面分析各个关键环节。文章提供刷新后钩子和启动后任务的代码示例&#xff0c;帮助读者理解SpringBoot源…...

deepseek v1手机端部署

在iPhone上部署DeepSeekR1 1. 安装快捷指令&#xff1a; 打开iPhone上的Safari浏览器&#xff0c;访问[这个链接](https://www.icloud.com/shortcuts/e0bc5445c39d45a78b90e1dc896cd010)下载快捷指令。 下载后&#xff0c;按照提示完成安装。 2. 获取并配置API Key&a…...

idea对jar包内容进行反编译

1.先安装一下这个插件java Bytecode Decompiler 2.找到这个插件的路径&#xff0c;在idea的plugins下面的lib文件夹内&#xff1a;java-decompiler.jar。下面是我自己本地的插件路径&#xff0c;以作参考&#xff1a; D:\dev\utils\idea\IntelliJ IDEA 2020.1.3\plugins\java-d…...

KMP算法原理 JAVA实现

KMP算法原理 JAVA实现 一、什么是KMP算法二、为什么需要KMP算法1. 算法背景1.1 暴力匹配过程1.2 暴力匹配的优劣 2. KMP算法的诞生3. next数组3.1 kmp算法的关键 三、求解KMP 一、什么是KMP算法 实际上KMP只是发明这个算法的三个人的英文名首字母短称&#xff0c;KMP本身无意义…...

利用Redis实现数据缓存

目录 1 为啥要缓存捏&#xff1f; 2 基本流程&#xff08;以查询商铺信息为例&#xff09; 3 实现数据库与缓存双写一致 3.1 内存淘汰 3.2 超时剔除&#xff08;半自动&#xff09; 3.3 主动更新&#xff08;手动&#xff09; 3.3.1 双写方案 3.3.2 读写穿透方案 3.3.…...

基于 RAMS 的数据驱动建模与应用实践:从理论到具体操作

基于 RAMS 的数据驱动建模与应用实践&#xff1a;从理论到具体操作 RAMS&#xff08;区域大气建模系统&#xff09;因其模块化设计、高分辨率模拟能力和广泛的应用领域&#xff0c;成为区域大气建模的强大工具。而数据驱动建模技术的崛起&#xff0c;使得 RAMS 的能力得到进一…...