腾讯2016研发工程师笔试题(一)—-两个值得注意的选择题
1,以下代码是否完全正确,执行可能得到的结果是__。
class A{
int i;
};
class B{
A *p;
public:
B(){p=new A;}
~B(){delete p;}
};
void sayHello(B b){
}
int main(){
B b;
sayHello(b);
}
A.程序正常运行
B.程序编译错误
C.程序崩溃
D.程序死循环
答案:C
分析:
为了清晰可见,我们从新把题目代码码一遍:
class A{
int i;
};
class B{
A *p;
public:
B(){p=new A;}
~B(){delete p;}
/*
B(const B& ths){
p = ths.p;
}*/
};
void sayHello(B x){
}
int main(){
B b;
sayHello(b);
}
这里的错误原因是编译器在生成default copy construction的时候使用的bitwise copy语义,也就是只是简单的浅拷贝。 上面被注释掉的程序就是编译器自动添加的部分。 从而导致在sayHello中向参数x传递值时,调用了bitwise copy的拷贝构造函数,使得x对象和b对象中的值完全一致,包括p指针的值,在x离开作用域(也就是sayHello函数结束),x发生析构,调用delete 销毁了指针p,同时在main函数结束的时候,析构b时又会调用一次delete删除指针p。
也就是本程序会delete一直已经被delete 的指针。可以做如下改进,来修复程序:
class A{
int i;
};
class B{
A *p;
public:
B(){p=new A;}
~B(){delete p;}
B(const B& other){
p = new A; //构建新的指针
*p = *(other.p); //将指向的内容复制,依然指向不同的位置
}
};
void sayHello(B b){
}
int main(){
B b;
sayHello(b);
}
如上,在B中添加copy 构造函数。
2,以下代码输出什么__.
main()
{
int a[5]={1,2,3,4,5};
int *p=(int *)(&a+1);
printf("%d",*(p-1));
}
A.1
B.2
C.5
D.出现错误
答案:C
分析:
&a表示一个指向大小为5数组的指针,那么&a+1就是表示一个指向大小为5的下一个数组的指针,也就是数组a最后一个元素的下一个位置,那么int*p=(int*)(&a+1)进行强制类型转换,将指向数组的指针转换为指向第二个数组中首元素的指针,所以p-1则是指向第一个数组中最后一个元素的指针,所以输出是5