亲宝软件园·资讯

展开

C++11 std::addressof 详解C++11的std::addressof源码解析

彼 方 人气:0
想了解详解C++11的std::addressof源码解析的相关内容吗,彼 方在本文为您仔细讲解C++11 std::addressof的相关知识和一些Code实例,欢迎阅读和指正,我们先划重点:C++11,std::addressof,下面大家一起来学习吧。

1、源码准备

本文是基于gcc-4.9.0的源代码进行分析,std::addressof是C++11才加入标准的,所以低版本的gcc源码是没有这个的,建议选择4.9.0或更新的版本去学习,不同版本的gcc源码差异应该不小,但是原理和设计思想的一样的,下面给出源码下载地址
http://ftp.gnu.org/gnu/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的处理过程可以分为以下四步:

4、总结

本文通过一个简单的例子和源码介绍了C++11新引入的模板函数std::addressof,内容虽然比较简单,但是考虑到std::addressof在标准库中使用频率是比较高的,所以我们还是应该掌握其原理和用法。

最后需要注意的一点就是std::addressof并不保证转换的正确性和合理性,这个是需要程序员自己把控的,虽然标准库使用std::addressof的频率比较高,但是我们平时在编程中还是得谨慎一点使用它。

加载全部内容

相关教程
猜你喜欢
用户评论