C++面试八股文:在C++中,你知道哪些运算符?
某日二师兄参加XXX科技公司的C++工程师开发岗位第11面:
面试官:在C++中,你都知道都哪些运算符?
二师兄:啥?运算符?
+-*/=
这些算吗?面试官:嗯,还有其他的吗?
二师兄:当然还有,
+=,-=,*=,/=,==
,还有逻辑运算,位运算等。面试官:好的。那你知道这些运算的优先级吗?
二师兄:(面试官傻逼吧,这谁记得住)记不住了,一般我都会加括号来表达我的意思。
面试官:好的。那你知道下面这段程序会输出什么吗?
#include <iostream>
int main(int argc, char const *argv[])
{
int i = 0;
std::cout << i++ + ++i << std::endl;
}
二师兄:应该是
2
吧。面试官:那你知道运算符的求值顺序吗?
二师兄:应该是从左向右?
面试官:
%
这个符号是求余的你应该知道吧。如果使用一个int
型负数对另一个int
型负数求余数,结果是正数还是负数?二师兄:应该是正数吧。
面试官:对一个整数判断是否位
true
或者false
,可以用if(val == true)
或if(val == false)吗?二师兄:不能,因为在使用
int
型与bool
型比较时,会把true
和false
转换成int
型,上面两个表达式等同于if(val == 1)
和if(val == 0),与我们的本意不符。面试官:你知道
a=a+1
和a+=1
这两者的区别吗?二师兄:两者应该没啥区别吧?
面试官:你知道
++i
和i++
的区别吗?二师兄:前者返回
i+1
,后者返回i
。之后i
会被+1
。面试官:C++11中的左值右值你熟悉吗?
二师兄:了解过一些(我很熟悉)。
面试官:那你知道算数运算符、逻辑运算符、位运算符返回的结果是左值还是右值吗?
二师兄:额,让我想想。。。应该都是右值。
面试官:好的。那你知道点(
.
)和箭头(->
)运算符返回的结果是左值还是右值吗?二师兄:额。。。应该都是左值。(应该是的吧。。。)
面试官:我们都知道,一个
int
型的负数,在最高(31
)位上的值是1
。那么如果对这个负数进行右移(>>
)操作,最高位上的1
会被移动吗?二师兄:应该不会。可能跟编译器实现有关。
面试官:你知道重载运算符要注意哪些事项吗?
二师兄:其实工作中很少重载运算符,我觉得最重要的事项就是尽量不要重载运算符。(我好机智。。)
面试官:好的,今天的面试结束了,回去等通知吧。
对于今日二师兄的表现,让我们来回过头看一下:
在C++中,你都知道哪些运算符?
除了算术运、逻辑、位、关系、等于运算符,如sizeof
,decltype
、new/delete
也属于运算符。
知道下面这段程序会输出什么吗?
应该是
2
吧。
这里程序的输出的结果是不确定的。原因会在下面讲。
那你知道运算符的求值顺序吗?
在C++的标准中,大部分的运算符两侧的表达式的求值顺序是不确定的。在上个例子中,可能会先计算i++,然后再计算++i,也可能反过来。所以这里的结果不缺行,属于未定义的行为(undefined behavior
)。
那么C++中有没有确定求值顺序的二元运算符呢?答案是有的,而且仅有这四个:&&
,||
,:?
,,
。C++标准保证这四个运算符的求值顺序是从左到右确定的。
你知道
a=a+1
和a+=1
这两者的区别吗?
此两者有一点区别,那就是前者的a
会被求值2
次,而后者只会被求值1
次。如果开启编译器优化,有可能会被优化相同的效果。
你知道
++i
和i++
的区别吗?
除了二师兄的回答外,前置++的效率要高于后者,因为前者不需要缓存值,以用来返回。
知道点(
.
)和箭头(->
)运算符返回的结果是左值还是右值吗?
这里箭头运算符的调用者肯定是左值(想想看为什么),但是号运算符的调用者可不一定是左值。所以箭头运算符的结果一定是左值,当点运算符的调用者是左值时,返回值时左值,当点运算符的调用者是右值时,返回值时右值。
负数进行右移(
>>
)操作,最高位上的1
会被移动吗
很遗憾,这里回答会和不会都不对,因为这也属于未定义的行为(undefined behavior
)。 所以不要对有符号的类型进行按位操作,最好的情况是正好当前编译器的实现和你的预期吻合,当前平台运行无异常。一旦移植代码,可能会遇到意想不到的问题。
你知道重载运算符要注意哪些事项吗?
二师兄机智!非必要不需要重载运算符。如果一定要重载,请注意:(我还是不建议你重载运算符,回头是岸。。。下面的规则都不用看了)
- 重载运算符必须至少有一个操作数是用户定义的类型,这意味着不能重载内置类型的运算符。
- 重载运算符必须具有与原始运算符相同的优先级和关联性。
- 重载运算符必须具有与原始运算符相同的参数数目。例如,二元运算符必须有两个参数,一元运算符必须有一个参数。
- 重载运算符不能更改运算符的含义,例如,不能将“+”运算符用于减法操作。
- 重载运算符必须是类的成员函数或全局函数。如果是成员函数,则第一个参数必须是类的对象。
- 重载运算符不能更改操作数的数量或类型。例如,不能将二元运算符重载为一元运算符。
- 重载运算符不能具有默认参数。
- 重载运算符的返回类型应该是运算符执行后的结果类型。
今日份面试到这里就结束了,小伙伴们,对于今天二师兄的表现能打几分呢?如果是你,以上的问题都能回答的上来吗?
关注我,带你21天“精通”C++!(狗头)