C++面试八股文之左值与右值

 更新时间:2023年06月18日 10:14:43   作者:二进制架构  
简单来说,左值就是可以使用&符号取地址的值,而右值一般不可以使用&符号取地址,这篇文章主要来和大家讲讲面试中左值与右值常考的知识点,需要的可以参考一下

某日二师兄参加XXX科技公司的C++工程师开发岗位第16面:

面试官:什么是左值,什么是右值?

二师兄:简单来说,左值就是可以使用&符号取地址的值,而右值一般不可以使用&符号取地址。

int a = 42;	//a是左值,可以&a
int* p = &a;
int* p = &42;	//42是右值,无法取地址

二师兄:一般左值存在内存中,而右值存在寄存器中。

int a = 42, b = 1024;
decltype(a+b);	//类型为右值,a+b返回的值存在寄存器中
decltype(a+=b);	//类型为左值,a+=b返回的值存储在内存中

二师兄:严格意义上分,右值分为纯右值(pvalue)和将亡值(xvalue)。C++中,除了右值剩余的就是左值。

42;				//纯右值
int a = 1024;
std::move(a);	//将亡值

面试官:C++98/03中已经有了左值,为什么还要增加右值的概念?

二师兄:主要是为了效率。特别是STL中的容器,当需要把容器当作参数传入函数时:

void function(std::vector<int> vi2)
{
    vi2.push_back(6);
    for(auto& i: vi2) { std:: cout < i << " " ;}
    std::cout << std::endl;
}
int main(int argc, char* argv[])
{
    std::vector<int> vi1{1,2,3,4,5};
    function(vi1);
    return 0;
}

二师兄:当我们要把vi1传入函数时,在C++98/03时只能通过拷贝构造函数,把vi1中所有的元素全部拷贝一份给vi2,拷贝完成之后,当function函数返回时,vi2被析构,然后vi1被析构。

二师兄:在C++11及之后,我们可以通过std::move()vi1强制转为右值,此时在初始化vi2时执行的不是拷贝构造而是移动构造:

void function(std::vector<int>&& vi2)
{
    vi2.push_back(6);
    for(auto& i: vi2) { std:: cout < i << " " ;}
    std::cout << std::endl;
}
int main(int argc, char* argv[])
{
    std::vector<int> vi1{1,2,3,4,5};
    function(std::move(vi1));
    return 0;
}

二师兄:这里只进行了一次构造。一次移动(当元素特别多时,移动的成本相对于拷贝基本可以忽略不记),一次析构。效率得到很大的提升。

二师兄:当然,移动过后的变量已经不能再使用(身体被掏空),在std::move(vi1)之后使用vi1是未定义行为。

面试官:好的。那你知道移动构造是如何实现的吗?

二师兄:移动构造是通过移动构造函数实现的,当类有资源需要管理时,拷贝构造会把资源复制一份,而移动构造偷走了原对象的资源。

struct Foo
{
    int* data_;
    //copy construct
    Foo(const Foo& oth)
    {
        data_ = new int(*oth.data_);
    }
    //move construct
    Foo(Foo&& oth) noexcept
    {
        data_ = oth.data_;		//steal
        oth.data_ = nullptr;	//set to null
    }
}

面试官:好的。你觉得移动构造函数的noexcept关键字能省略吗?为什么?

二师兄:应该不能吧,具体不清楚。

面试官:那你知道std::move是如何实现的吗?

二师兄:好像是static_cast实现的吧。

面试官:那你知道什么叫万能引用吗?

二师兄:万能引用主要用在模板中,模板参数是T,形参是T&&,此时可以传入任何类型的参数,所以称之为万能引用。

template<typename T>
void function(T&& t) { ...}

面试官:那你知道万能引用是如何实现的吗?

二师兄:不太清楚。。

面试官:完美转发知道吗?

二师兄:std::forward 吗,了解过一些,不太熟悉。

面试官:好的,回去等消息吧。

让我们来回顾以下二师兄今天的表现:

移动构造函数的noexcept关键字能省略吗?为什么?

这里尽量不要省略。如果省略,编译器会推断是否会抛出异常。如果移动构造函数可能会抛出异常,则编译器不会将其标记为noexcept。当编译器不标记为noexcept时,为了保证程序的正确性,编译器可能会采用拷贝构造的方式实现移动构造,从而导致效率降低。

需要注意的是,如果标记了noexcept但在移动时抛出了异常,则程序会调用std::terminate()函数来终止运行。

知道std::move是如何实现的吗?

这里的确是通过static_cast实现的,讲左值强行转换成右值,用来匹配移动语义而非拷贝。

template&lt;typename T&gt;
typename std::remove_reference&lt;T&gt;::type&amp;&amp; move(T&amp;&amp; t) { return static_cast&lt;typename std::remove_reference&lt;T&gt;::type&amp;&amp;&gt;(t);}

万能引用是如何实现的?

万能引用主要使用了引用折叠技术,

template<typename T>
void function(T&& t) { ...}

当T类型为左值时,&& & 被折叠为&, 当T类型为右值时,&& &&被折叠称为&&。以下是折叠规则:

& &    -> &
& &&   -> &
&& &   -> &
&& &&  -> &&

完美转发知道吗?

当我们需要在function中传递t参数时,如何保证它的左值或右值语义呢?这时候完美转发就登场了:

template<typename T>
void function2(T&& t2) {}
template<typename T>
void function(T&& t) 
{
    function2(t);
}

当传入的参数t的类型时右值时,由于引用折叠还是右值,此时的t虽然时一个右值引用,但t本身却是一个左值!这里非常的不好理解。如果我们把t直接传入到function2,那么function2中的t2会被推导成左值,达不到我们的目标。如果在调用function2时传入std::move(t),当t是右值时没有问题,但当t是左值时,把t移动到t2t在外部不在能用。这也不符合我们的预期。此时std::forward闪亮登场!

template<typename T>
void function2(T&& t2) {}
template<typename T>
void function(T&& t) 
{
    function2(std::forward<T&&>(t));
}

std::forward使用了编译时多态(SFINAE)技术,使得当参数t是左值是和右值是匹配不同的实现,完成返回不同类型引用的目的。以下是标准库的实现:

template <typename _Tp>
constexpr _Tp && forward(typename std::remove_reference<_Tp>::type &&__t) noexcept
{
    return static_cast<_Tp &&>(__t);
}
template <typename _Tp>
constexpr typename std::remove_reference<_Tp>::type && move(_Tp &&__t) noexcept
{
    return static_cast<typename std::remove_reference<_Tp>::type &&>(__t);
}

好了,今日份面试到这里就结束了。二师兄的表现如何呢?预知后事如何,且听下回分解。

到此这篇关于C++面试八股文之左值与右值的文章就介绍到这了,更多相关C++左值右值内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C++ vector的简单实现

    C++ vector的简单实现

    这篇文章主要为大家详细介绍了C++ vector的简单实现,使用数据库,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03
  • C/C++ 函数的存储位置和占用空间详解

    C/C++ 函数的存储位置和占用空间详解

    Lambda函数的代码部分在代码段中,被捕获的变量存储在Lambda函数对象的内部,这些变量的存储位置取决于Lambda函数对象的存储位置,这篇文章主要介绍了C/C++函数的存储位置和占用空间,需要的朋友可以参考下
    2023-06-06
  • 判断整数序列是否为二元查找树的后序遍历结果的解决方法

    判断整数序列是否为二元查找树的后序遍历结果的解决方法

    本篇文章是对判断整数序列是否为二元查找树的后序遍历结果的解决方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • 关于C++中push_back()函数的用法及代码实例

    关于C++中push_back()函数的用法及代码实例

    push_back是vector的一个方法,表示将一个元素存储到容器的末尾,下面这篇文章主要给大家介绍了关于C++中push_back()函数用法的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • C++数据结构之并查集详解

    C++数据结构之并查集详解

    这篇文章主要介绍了C++数据结构之并查集详解,并查集是一种树型的数据结构,用于处理一些不相交集合的合并及查询问题,并查集的思想是用一个数组表示了整片森林,需要的朋友可以参考下
    2023-08-08
  • C语言函数超详细讲解下篇

    C语言函数超详细讲解下篇

    函数是一组一起执行一个任务的语句。每个 C 程序都至少有一个函数,即主函数 main() ,所有简单的程序都可以定义其他额外的函数,函数我们分两篇来讲解,接下来开始第二篇
    2022-04-04
  • C语言单链表实现学生管理系统

    C语言单链表实现学生管理系统

    这篇文章主要为大家详细介绍了C语言单链表实现学生管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-12-12
  • C语言数据结构之图的遍历实例详解

    C语言数据结构之图的遍历实例详解

    这篇文章主要介绍了C语言数据结构之图的遍历实例详解的相关资料,需要的朋友可以参考下
    2017-07-07
  • vscode配置gitbash终端的方法步骤

    vscode配置gitbash终端的方法步骤

    本文主要介绍了vscode配置gitbash终端的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-06-06
  • C++设计模式中的观察者模式一起来看看

    C++设计模式中的观察者模式一起来看看

    这篇文章主要为大家详细介绍了C++观察者模式,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03

最新评论