温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

C++11中的std::addressof怎么用

发布时间:2022-04-14 17:01:39 来源:亿速云 阅读:162 作者:zzz 栏目:编程语言

本文小编为大家详细介绍“C++11中的std::addressof怎么用”,内容详细,步骤清晰,细节处理妥当,希望这篇“C++11中的std::addressof怎么用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

1、源码准备

本文是基于gcc-4.9.0的源代码进行分析,std::addressof是C++11才加入标准的,所以低版本的gcc源码是没有这个的,建议选择4.9.0或更新的版本去学习,不同版本的gcc源码差异应该不小,但是原理和设计思想的一样的。

2、std::addressof简介

std::addressof的作用是获取一个对象的实际地址,即使这个对象的&操作符已被重载。它接受一个参数,该参数为要获得地址的那个对象的引用。下面通过一个极其简单的例子了解一下std::addressof的使用方法

#include <iostream>
#include <string>
#include <functional>

class Test
{
public:
    int* operator&()
    {
        return &b;
    }

    int* a_addr()
    {
        return &a;
    }

    int* b_addr()
    {
        return &b;
    }

private:
    int a;
    int b;
};

int main(int argc, char* argv[])
{
    Test t;
    std::cout << "&t.a:" << t.a_addr() << std::endl;
    std::cout << "&t.b:" << t.b_addr() << std::endl;
    std::cout << "&t:" << &t << std::endl;
    std::cout << "addressof(t):" << std::addressof(t) << std::endl;
}

上面的代码输出结果为:

&t.a:0x7ffefcb48eb0
&t.b:0x7ffefcb48eb4
&t:0x7ffefcb48eb4
addressof(t):0x7ffefcb48eb0

在这里正常来说使用&t应该取到的是t.a的地址值才对,但是由于我们重载了&运算符,所以这里取到了t.b的地址值,但是如果使用了std::addressof,就可以取到正确的值了。

3、std::addressof源码解析

std::addressof位于libstdc++-v3\include\bits\move.h中

template<typename _Tp>
inline _Tp* __addressof(_Tp& __r) _GLIBCXX_NOEXCEPT
{
    return reinterpret_cast<_Tp*>(&const_cast<char&>(reinterpret_cast<const volatile char&>(__r)));
}

template<typename _Tp>
inline _Tp* addressof(_Tp& __r) noexcept
{ return std::__addressof(__r); }

从代码中可以看出std::addressof里面调用了std::__addressof,所以真正起作用的是std::__addressof。__addressof的处理过程可以分为以下四步:

  • 将__r由类型_Tp&强制转换为const volatile char&,这样做有两个作用:一是防止后面使用&操作符获取地址时触发原类型(即_Tp)的重载操作(operator&),就像上面那个例子那样;二是reinterpret_cast操作符总是可以合法的在原类型的基础上加const或volatile, 但是如果_Tp原来就带有const或volatile的话, 通过reinterpret_cast去掉是不允许的, 因此需要加上const volatile来避免编译器报错, 也就是此时不用再管_Tp是否本来就带有const或volatile属性了。

  • 将前面转换得到的结果强制转换为char&类型,此时如果转换成其它类型有可能会触发强制地址对齐的操作,这样的话真实地址就有可能会被改变了,最终造成程序错误。需要注意的是这个转换过程使用的是const_cast,可以顺便将前面留下的const和volatile属性给去掉了。

  • 使用&符号将前面的结果的地址给取出来(此时已经不会触发重载了)

  • 最后一步使用reinterpret_cast将前面获取到的地址转换回_Tp*类型,并且此时也会保留_Tp的const或volatile属性(如果有的话)

读到这里,这篇“C++11中的std::addressof怎么用”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注亿速云行业资讯频道。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI