面经:C++篇(持续更新)

一、 左值和右值

L-value中的L指的是Location,表示可寻址。Avalue (computer science)that has an address.
R-value中的R指的是Read,表示可读。in computer science, a value that does not have an address in a computer language.
左值和右值是相对于赋值表达式而言的。左值是能出现在赋值表达式左边的表达式。左值表达式可以分为可读写的左值和只读左值。右值是可以出现在赋值表达式右边的表达式,他可以是不占据内存空间的临时量或字面量,可以是不具有写入权的空间实体。如
int a=3;
const int b=5;
a=b+2; //a是左值,b+2是右值
b=a+2; //错!b是只读的左值但无写入权,不能出现在赋值符号左边
(a=4)+=28; //a=4是左值表达式,28是右值,+=为赋值操作符
34=a+2; //错!34是字面量不能做左值
 

二、左值引用和右值引用

左值引用

用法:Type & 左值引用名 = 左值表达式;

注意点:声明时必须初始化,初始化之后无法在改变;对别名的一切操作都等价于对原来变量的操作

右值引用:

int &rb = a + 1; 这样的语法,因为a + 1 此时是作为一个右值来使用的,我们无法把一个右值赋值给一个左值引用。左值引用相当于把一个变量的地址付给另一个变量,这两个变量可以访问同一个内存,右值仅仅是一个数,而非内存中的某块地址,因此无法把右值复制给左值引用(这句话并不准确,有的右值也是有地址的,比如字符串常量)。

声明方法:Type && 右值引用名 = 右值表达式;

T&& t在发生自动类型推断的时候,它是未定的引用类型(universal references),如果被一个左值初始化,它就是一个左值;如果它被一个右值初始化,它就是一个右值,它是左值还是右值取决于它的初始化。

三、拷贝构造函数

拷贝构造函数是C++独有的,它是一种特殊的构造函数,用基于同一类的一个对象构造和初始化另一个对象。

当没有重载拷贝构造函数时,通过默认拷贝构造函数来创建一个对象

A a;

A b(a); //调用拷贝构造函数

A b=a;  //调用拷贝构造函数

强调:这里b对象是不存在的,是用a 对象来构造和初始化b的!!

先说下什么时候拷贝构造函数会被调用:

在C++中,3种对象需要复制,此时拷贝构造函数会被调用

1)一个对象以值传递的方式传入函数体

2)一个对象以值传递的方式从函数返回

3)一个对象需要通过另一个对象进行初始化

T function1(){
2     T t(0);
3     return t;
4 }
5 T x=function1();

这里的过程是:
1.创建命名对象t
2.拷贝构造一个无名的临时对象,并返回这个临时对象
3.由临时对象拷贝构造对象x
4.T x=function1();这句语句结束时,析构临时对象
这里一共生成了3个对象,一个命名对象t,一个临时对象作为返回值,一个命名对象x。

四、 模板特化

1.C++模板:

//模板函数
template<typename T, class N> void func(T num1, N num2)
{
    cout << "num1:" << num1 << ", num2:" << num2 <<endl;
}
 
//模板类
template<typename T, class N> class Test_Class
{
    static bool comp(T num1, N num2)
    {
        return (num1<num2)?true:false;
    }
};

2. C++模板全特化

在模板类里,所有的类型都是模板(template<class T>),而一旦我们将所有的模板类型T都明确化,并且写了一个类名与主模板类名相同的类,那么这个类就叫做全特化类。C++模板全特化之后已经失去了Template的属性了。

//模板函数
template<typename T, class N> void func(T num1, N num2)
{
    //cout << "num1:" << num1 << ", num2:" << num2 <<endl;
}
 
//模板类
template<typename T, class N> class Test_Class
{
public:
    static bool comp(T num1, N num2)
    {
        return (num1<num2)?true:false;
    }
};
 
//全特化,模板函数
template<> void func(int num1, double num2)
{
    cout << "num1:" << num1 << ", num2:" << num2 <<endl;
}
 
//全特化,模板类
template<> class Test_Class<int, double>
{
public:
    static bool comp(int num1, double num2)
    {
        return (num1<num2)?true:false;
    }

注意:一个模板被称为全特化的条件:1.必须有一个主模板类  2.模板类型被全部明确化

3. C++模板偏特化

偏特化就是介于二者之间的模板,它的模板名与主版本模板名相同,但是它的模板型中,有被明确化的部分和没有被明确化的部分。

//模板函数
template<typename T, class N> void func(T num1, N num2)
{
    //cout << "num1:" << num1 << ", num2:" << num2 <<endl;
}
 
//模板类
template<typename T, class N> class Test_Class
{
public:
    static bool comp(T num1, N num2)
    {
        return (num1<num2)?true:false;
    }
};
 
//偏特化,模板函数
template<class N> void func(int num1, N num2)
{
    cout << "num1:" << num1 << ", num2:" << num2 <<endl;
}
 
//偏特化,模板类
template<class N> class Test_Class<int, N>
{
public:
    static bool comp(int num1, double num2)
    {
        return (num1<num2)?true:false;
    }

注意:偏特化的条件:1.必须有一个主模板   2.模板类型被部分明确化

4. 模板类调用优先级

全特化类>偏特化类>主版本模板类

模板特化作用:性能优化,针对某一类型特定实现,使模板能够正常工作

五、 STL 算法对自定义类型重载=运算符

六、 delete 和 delete[] 区别

 delete和delete[]对基本数据类型没有区别

 但对类对象有区别,delete只调用一次析构函数,而delete[]调用多次析构函数

七、继承

基类不能被继承的:构造函数,析构函数,赋值运算符,友元函数

不能被声明虚函数的:构造函数,new ,delete(因为new和delete是静态成员函数,静态成员函数是不能被声明为virtual的,因为virtual是针对对象,而静态是针对类)

默认生成的:构造函数,析构函数,赋值运算符=,取地址运算符&

6.shared_ptr内部实现原理

7.vector内存进制,插入,删除

8.内存管理方法

原文地址:https://www.cnblogs.com/dj0325/p/9484098.html