2404C++,C++ADL扩展库

2024-04-02 12:12
文章标签 c++ 扩展 adl 2404

本文主要是介绍2404C++,C++ADL扩展库,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文

ADL:依赖参数查找,是一组对函数调用式(包括隐式函数调用重载符号)中,未限定函数名查找的规则.

NS::foo(a); //限定
foo(a); //未限定

如果加了限定就是限定,比如加了名字空间类名去调用foo.没有限定的foo才会应用ADL查找规则.
cppreference中的一例来看ADL:

#include <iostream>
int main()
{std::cout << endl; //`#1`错误:`'endl'`未在此名字空间中声明.这不是调用`endl()`函数,所以不适用`ADL`endl(std::cout); //`#2`,OK:编译器会从参数`std::cout`所在的名字空间`std`中查找`endl`函数,并找到了`std::endl`,成功调用(endl)(std::cout); //`#3`错误:子式`(endl)`不是函数调用式,不会应用`ADL`
}

#1代码会编译错误,因为找不到endl,也不会应用ADL,因为endl不是个函数调用;
#2代码则是可调用的,因为编译器会从std::cout参数所在的std名字空间中查找endl函数,并找到了std::endl,成功调用.
#3代码会编译报错,因为(endl)式不是函数调用,不会应用ADL.

ADL更多的细节可参考

如何利用ADL规则来扩展现有库.前几天社区有个同学提了一个问题:这里,说struct_json无法转换如下结构为json串:

#include <eigen3/Eigen/Dense>
struct person {std::string name;int age;Eigen::Vector3d pos;
};
REFLECTION(person, name, age, pos);
int main() {person p{"tom", 20};std::string str;//`struct`到`json`struct_json::to_json(p, str);std::cout << str << "\n";person p1;//`json`到`struct`struct_json::from_json(p1, str);assert(p == p1);
}

from_jsonto_json时会编译报错,原因是因为Eigen::Vector3d第三方结构,它不是一个struct_json可识别的反射类型,而Eigen::Vector3d因为是第三方的结构,不方便转换为struct_json识别的反射类型.

如果要支持该类型,或在struct_json内部增加Eigen::Vector3d结构的序化和反序化,但该方法治标不治本,因为还可能有无数自定义类型,不可能每次都修改struct_json的代码.

更好方法:用ADL来扩展struct_json序化和反序化,允许用户自定义函数序化和反序化自定义或第三方结构.

struct_json::to_json内部调用了void to_json_impl(Stream&s,T&&t)具体来序化各种类型,只要让to_json_impl支持ADL,则就可在用户名字空间里序化json了.

先把限定sturct_json内部的去掉,之前用detail::voidto_json_impl来序化,现在去掉限制,改成未限定函数调用即可:

using namespace detail;
to_json_impl(s, t);

然后再在用户名字空间中定义to_json_impl:

namespace Eigen
{template <typename Stream>inline void to_json_impl(Stream &ss, Eigen::Vector3d const& val) {std::stringstream ss2;ss2 << "[" << val[0] <<"," << val[1] <<"," <<val[2] <<"]";ss.append(ss2.str());}
}

此时再调用之前的to_json时,根据ADL规则,编译器会根据Eigen::Vector3dconst&val参数,查找Eigen名字空间中的to_json_impl,并调用.

这样struct_json就可支持自定义类型,from_json类似.

利用ADL规则很优雅的扩展了库,没有侵入式修改也没有继承,是个很好的扩展库的方法.

这篇关于2404C++,C++ADL扩展库的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

PostgreSQL的扩展dict_int应用案例解析

《PostgreSQL的扩展dict_int应用案例解析》dict_int扩展为PostgreSQL提供了专业的整数文本处理能力,特别适合需要精确处理数字内容的搜索场景,本文给大家介绍PostgreS... 目录PostgreSQL的扩展dict_int一、扩展概述二、核心功能三、安装与启用四、字典配置方法

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性: