远程过程调用-buttonrpc源码解析7-函数返回值

2024-04-04 00:28

本文主要是介绍远程过程调用-buttonrpc源码解析7-函数返回值,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前一篇文章讲述了服务端如何进行函数调用,但并未分析函数返回值,本节进行重点分析。
在负责拆分委托函数的callproxy_函数内,最终得到了型如
typename type_xx<R>::type r = call_helper<R>(ff, args);或者
typename type_xx<R>::type r = call_helper<R>(func, args);的调用形式。
1、函数call_helper相关的定义如下:

// 定义一个泛化的特征模板,用于确定函数返回值类型:T型的type就是T本身
template<typename T>
struct type_xx{	typedef T type; };
// 定义一个特化模板:void型的type为int8_t
template<>
struct type_xx<void>{ typedef int8_t type; };// 调用帮助类,主要用于返回是否void的情况:是,并返回0
template<typename R, typename F, typename ArgsTuple>
typename std::enable_if<std::is_same<R, void>::value, typename type_xx<R>::type >::type
call_helper(F f, ArgsTuple args) {invoke(f, args);return 0;
}
// 调用帮助类,主要用于返回是否void的情况:否,返回真正的返回值
template<typename R, typename F, typename ArgsTuple>
typename std::enable_if<!std::is_same<R, void>::value, typename type_xx<R>::type >::type
call_helper(F f, ArgsTuple args) {return invoke(f, args);
}// 具体实现函数模板:用tuple做参数调用函数模板类
template<typename Function, typename Tuple, std::size_t... Index>
decltype(auto) invoke_impl(Function&& func, Tuple&& t, std::index_sequence<Index...>)
{return func(std::get<Index>(std::forward<Tuple>(t))...);
}
// 函数模板:作为一个接口,负责调用相关函数并传递参数
template<typename Function, typename Tuple>
decltype(auto) invoke(Function&& func, Tuple&& t)
{constexpr auto size = std::tuple_size<typename std::decay<Tuple>::type>::value;return invoke_impl(std::forward<Function>(func), std::forward<Tuple>(t), std::make_index_sequence<size>{});
}

首先定义一个特征类模板type_xx用来获得函数的返回值类型,这里void类型的返回值并非void,而是特化为int8_t类型,是因为无法定义一个void类型的变量来接call_helper的返回值。
接着定义了两个call_helper函数模板,通过使用std::enable_if来利用函数模板的重载决议SFINAE特性,为void类型和其它类型实例化不同的函数。通过decltype(auto),能够自动推导函数返回值类型。
最后定义invokeinvoke_impl两个函数模板,这两个函数模板用来执行函数的实际调用,之前的文章有讲过,这里不再赘述。
其实到这里,整个服务端函数调用的过程差不多已经分析完了,invoke函数已经能够得到正确的返回值了,而call_helper函数也已完成了它的使命,给出了对应的返回值。
温馨提示std::is_same用于比较两个类型是否一致,类型一致则value为true,否则value为false。std::enable_if用于条件编译,它通常与函数模板一起使用,以便根据某些类型特性或表达式结果来选择合适的函数重载(常出现于SFINAE场景中)。decltype(auto)主要用于推导转发函数和类似包装的返回类型。
2、函数返回值包裹类型
函数调用成功与否,都需要给出反馈信息,至少包括状态码(通常为枚举类型)、错误信息(通常为string类型)和返回值。我们可以将这些信息依次序列化到Serializer对象,也可以将这些信息进行打包,形成一个独立的类,buttonrpc中的包裹返回值类:

template<typename T>
class value_t {
public:typedef typename type_xx<T>::type type;typedef std::string msg_type;typedef uint16_t code_type;value_t() { code_ = 0; msg_.clear(); }bool valid() { return (code_ == 0 ? true : false); }int error_code() { return code_; }std::string error_msg() { return msg_; }type val() { return val_; }void set_val(const type& val) { val_ = val; }void set_code(code_type code) { code_ = code; }void set_msg(msg_type msg) { msg_ = msg; }friend Serializer& operator >> (Serializer& in, value_t<T>& d) {in >> d.code_ >> d.msg_;if (d.code_ == 0) {in >> d.val_;}return in;}friend Serializer& operator << (Serializer& out, value_t<T> d) {out << d.code_ << d.msg_ << d.val_;return out;}
private:code_type code_;msg_type msg_;type val_;
};

这篇关于远程过程调用-buttonrpc源码解析7-函数返回值的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/874369

相关文章

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

MySQL count()聚合函数详解

《MySQLcount()聚合函数详解》MySQL中的COUNT()函数,它是SQL中最常用的聚合函数之一,用于计算表中符合特定条件的行数,本文给大家介绍MySQLcount()聚合函数,感兴趣的朋... 目录核心功能语法形式重要特性与行为如何选择使用哪种形式?总结深入剖析一下 mysql 中的 COUNT

一文详解Git中分支本地和远程删除的方法

《一文详解Git中分支本地和远程删除的方法》在使用Git进行版本控制的过程中,我们会创建多个分支来进行不同功能的开发,这就容易涉及到如何正确地删除本地分支和远程分支,下面我们就来看看相关的实现方法吧... 目录技术背景实现步骤删除本地分支删除远程www.chinasem.cn分支同步删除信息到其他机器示例步骤

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

MySQL中的InnoDB单表访问过程

《MySQL中的InnoDB单表访问过程》:本文主要介绍MySQL中的InnoDB单表访问过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、访问类型【1】const【2】ref【3】ref_or_null【4】range【5】index【6】