每天一点C++——杂记
结构体的深拷贝和浅拷贝
浅拷贝就是只拷贝指针,并不拷贝指针所指向的内容,深拷贝则会对指针的内容进行拷贝。浅拷贝会在一些场景下出现问题,看下面的例子:
struct s {
char * name;
int age;
};
如果我定义 一个对象s1,并且为name申请一块内存并赋值为“zhangsan”,然后定义对象s2=s1,然后修改s2的name值会发生什么情况?
重载实现原理
重载可以让相同名称的函数具有不同的实现,可以通过参数的数量,参数类型,参数顺序等来区分,但是不能够只通过返回值来判断。
重载是原理就是基于编译器name Mangling机制,在编译阶段会为所有函数生成一个唯一的标识符用来区分不同函数。通过编译器将具有相同名称不同参数的函数会根据名称加参数生成唯一的标识符,并在执行时调用,这也叫静态联编。
这里的name mangling 机制是:前缀 _Z是GCC的规定(写gcc编译器人员规定的),数字3 是函数名Add的字符个数,i表示第一个函数的参数类型int,i是第二个函数的参数类型int。对于第二个函数签名中Pi则表示第三个函数类型是int*
因为C编译器的name mangling规则与C++的不同:C语言的命名规则仅依赖于函数名,和函数参数类型无关。因为同名函数没办法区分。而C++是通过函数修饰规则来区分,只要参数不同,修饰出来的名字就不一样,就支持了重载。
const
const用来标记一个变量是不可修改的,主要使用的场景有:
- 修饰变量
- 修饰指针
- 修饰引用
- 修饰函数参数
- 修饰函数返回值
- 修饰类成员函数
对应的还要一个constexpr 表示这个一个常量表达式,可以在编译阶段计算出来
指针和引用的区别
- 引用在声明时必须赋值,指针不需要
- 引用赋值后不可以修改,指针可以
- 引用不可以为空值,指针可以
虚函数、纯虚函数、抽象类
虚函数
虚函数并不是不实现的函数,而是允许基类指针可以调用子类的该函数。用来实现多态,虚函数只能借助于指针或者引用来达到多态的效果。
纯虚函数
如果一个函数为纯虚函数,则代表该函数没有实现,只是定义了一个接口,规定继承该类的子类必须实现该函数。
抽象类
带有纯虚函数的类称为抽象类。抽象类不能够定义对象,如果子类不是先纯虚函数则子类也是一个抽象类。
虚继承
解决菱形继承的问题。解决命名冲突和冗余数据问题, 使得在派生类中只保留一份间接基类的成员。
四种cast
reinterpret_cast
reinterpret_cast是功能最强大的cast,但也是最暴力,最底层,最不安全的,他是一个编译器指令,将对应的地址转换解释的方式。
可以将一个int转换为指针,也可以将指针转换为int,也可以实现不同类型指针之间的转换。
reinterpret_cast是底层二进制的强制拷贝和语义转换不会损失精度。
const_cast
修改变量的const属性。
- 如果是添加const,则是先进行一个深拷贝然后赋值
- 如果是去掉const
static_cast
- 基本类型之间转换
- void指针转换为任意基本类型指针
- 子类或父类之间引用或指针的转换,向下转换父类转子类是不安全的,子类转父类是安全的
dynamic_cast
dynamic_cast用于类继承层次间的指针或引用转换(主要用于向下的安全转换)。dynamic_cast向下转型的安全性主要体现在RTTI运行时类型识别。无法转型返回NULL。
auto_ptr, unique_ptr, shared_ptr, weak_ptr
只能指针用于解决内存正确释放的问题,它负责自动释放所指向的对象
auto_ptr:
unique_ptr
某个时刻只能有一个unique_ptr指向一个给定对象。当unique_ptr 被销毁时,它所指向的对象也被销毁。
初始化
没有shared_ptr中make_shared标准库函数返回一个unique_ptr,我们定义一个unique_ptr需要绑定到new返回的指针上,并且不能通过=赋值的方式,要采用直接赋值方式,unique_ptr p1(new A()).
release,reset
禁止拷贝和赋值
因为该指针指向的对象只能有一个unique_ptr指针,也就是一个引用计数, 因此不支持普通的拷贝和赋值。特殊情况,一个即将销毁的unique_ptr可以拷贝或者赋值给另一个unique指针。
shared_ptr
作用
消除野指针,重复释放,内存泄漏等问题。
基本原理
记录对象的引用次数,当引用次数为0的时候,也就是最后一个共享指针析构的时候,会把指向的内存释放掉。
常用函数
- get
- use_count
- reset
- operator*
- operator->
注意事项
- 多线程的读写安全
- 要谨慎使用get
- 不要用栈内的指针构造shared ptr,因为共享指针是通过delete来释放内存,需要对应到new出的内存才可以
- 不要用原始指针初始化多个共享指针
weak_ptr:
创建weak_ptr, 需要使用shared_ptr,反过来是行不通的,需要使用lock函数; weak_ptr不会增加引用计数; 解决shared_ptr循环引用,将其中一个改为weakptr即可
内联函数
inline关键字修饰的函数或者是函数内部定义的函数都可能是内联函数,但是具体会不会展开还需要编译器决定。
- 内联函数是以代码膨胀为代价的,如果内联函数内有循环或者代码较长,执行时间比调用时间更长
- 代码膨胀后会消耗更多的内存