C++面向对象高级编程(九)Reference与重载operator new和operator delete
摘要: 技术在于交流、沟通,转载请注明出处并保持作品的完整性。
一 Reference
引用:之前提及过,他的主要作用就是取别名,与指针很相似,实现也是基于指针.
1.引用必须有初值,且不能引用nullptr
2.引用之后不能再引用别人
3.引用通常不用于声明变量,多用于参数类型,和返回值类型
见下面代码
int main(int argc, const char * argv[]) {
int x=;
//p is a pointer to x
int* p = &x;
// r is a reference to x
int& r = x;
//这个操作并不是r引用了别人而是 是r引用的x = 5
int x2 = ;
r = x2; // x = 5
cout << "x = "<< sizeof(x) << endl;
cout << "r = "<< sizeof(r) << endl;
cout << "r的地址 = "<< &r <<endl;
cout << "p的地址 = "<< p <<endl;
return ;
}
输出结果

你会发现 r的地址个p的地址是相同的,其实引用并没有实际的内存,为了实现引用的取别名的假象,编译器会这么做 sizeof(r) = sizeof(x)和 &x = &r
4.引用是指针,是一种漂亮的指针
void func2 (string* obj) { obj ->clear();} //pass by pointer
void func1 (string obj) { obj .clear();} //pass by value
void func3 (string& obj) { obj .clear();} //pass by reference
调用端
string a;
//pass by pointer 接口不同
func2(&a); //pass by value 调用接口相同 效率低
func1(a); //pass by reference 调用接口相同
func3(a);
5.不能用引用区分函数签名
double imag(const double& im) {return im;};
double imag(const double im) {return im;};
调用的时候你会发现

错误提示是函数调用混淆,你会发现不能用引用区分函数签名(上面红色部分为函数签名)
const也算函数签名的一部分
二 多态
带vaiturl 内存上会多一个4字节的指针
继承是继承调用权 而不是继承内存大小
动态绑定的三个前提
1.必须通过指针调用
2.指针为向上的关系(保证安全)
3.调用的是虚函数
(*(p->vptr)[n])(p));
(*p->vptr[n])(p)
三 const
a.const object(data member不得改变)
b.non-const object(data member可以改变)
c.const member function(保证不改变data member)
d.non-const member functions(不保证 data member不变)
他们的组合关系

当成员函数的const 和non-const版本同时存在,const object 只会调用const 版本,non-const object 只会调用non-const
我们思考一下
const String str("Hello world");
str.print();
如果print()不加const[const object与non-const member function] 会报错
Copy On Write,带const 不用考虑 COW
class template std::basic_string<...>
//里面有两个member function
charT operator[] (size_type pos) const {不考虑COW == Copy On Write};
reference operator[](size_type pos){必须考虑COW};
四 New 和 Delete
之前提过
new = malloc() + ctor
delete = dtor + free()
new 和delete不可以重载 但是 operator new 和 operator delete可以重载
class Foo
{
public: Foo()
{
cout<< "Foo ("<< ++ctorNum << ")"<<endl;
}
~Foo()
{
cout<< "~Foo ("<< ++dtorNum << ")"<<endl;
}
//重载成员operatpr new
void* operator new(size_t size) noexcept
{
cout<< "Foo operator new" <<endl; return malloc(size);
}
//重载成员operatpr delete
void operator delete(void* ptr, size_t size)
{
cout<< "Foo operator delete" <<endl;
free(ptr);
}
}
调用端
int main()
{
Foo * p = new Foo;
delete p;
}
结果

发现我们接管了 operator new 和operator delete
class Foo
{
public: Foo()
{
cout<< "Foo ("<< ++ctorNum << ")"<<endl;
}
~Foo()
{
cout<< "~Foo ("<< ++dtorNum << ")"<<endl;
}
//重载成员operatpr new
void* operator new[](size_t size)
{
cout<< "Foo operator new" <<endl; return malloc(size);
} //重载成员operatpr delete[]
void operator delete[](void* ptr, size_t size)
{
cout<< "Foo operator delete[]"<<endl;
cout << size <<endl;
free(ptr);
} static int ctorNum;
static int dtorNum;
}; int Foo::ctorNum = ;
int Foo::dtorNum = ;
调用端
int main()
{
Foo* l = new Foo[];
delete[] l;
return ;
}
结果

我们可以重载多个版本的class menber operator new() 第一参数必须为 size_t
我们可以重载 class member operator delete() 可以重载过个版本 ,但是他们绝对不会被delete
调用重载的delete()只有当new所调用的ctor抛出 exception 才会调用这些重载版的operator delete() 它只可能这样被调用,主要用归还未能完全城建成功的object所占用的 内存
调用会一一对应 ,如果你的delete()与new() 没有一一对应编译器 也不会报错 表示放弃对异常做处理放弃
class Bad{};//抛出异常用
class test
{
public:
test(){}
test(int i){/*故意抛出异常*/ throw Bad();}
~test(){}
//一般的operator new()
void* operator new(size_t size){cout<< "一般的operator new()" <<endl; return malloc(size);}
//一般的operatpr delete
void operator delete(void* ptr, size_t size){cout<< "一般的operator delete()"<<endl; free(ptr);}
//标准库的operator new()
void* operator new(size_t size, void* start){cout<< "标准库的operator new()" <<endl; return start;}
//标准库的operator delete()
void operator delete(void*, void*){cout<< "标准库operator delete(void*, void*)" <<endl;}
//崭新的operator new()
void* operator new(size_t size, long extra){cout<< "崭新operator new()" <<endl; return malloc(size + extra);}
//崭新的operator new()
void operator delete(void*, long){cout<< "崭新operator delete(void*, long)" <<endl;}
//又一个operator new()
void* operator new(size_t size, long extra, char init){cout<< "又一个operator new()" <<endl; return malloc(size + extra);}
//又一个operator delete()
void operator delete(void* size, long extra, char init){cout<< "又一个operator delete()" <<endl;}
};
调用及输出
int main()
{
try{
test t;
test* p1 = new test;
test* p2 = new (&t) test;
test* p3 = new() test;
test* p4 = new(,'a') test;
test* p5 = new() test();
test* p6 = new(,'a') test();
test* p7 = new(&t) test();
}
catch(...){};
return ;
}

发现new和delete是一一对应,至于其他的为什么没有打印出来是由于,new不成功抛出异常导致
完!
参照<<侯捷 C++面向对象高级编程>>
C++面向对象高级编程(九)Reference与重载operator new和operator delete的更多相关文章
- C++面向对象高级编程(二)基础篇
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 概要 知识点1.重载成员函数 知识点2 . return by value, return by reference 知识点3 重载非成员函数 ...
- C++面向对象高级编程(一)基础篇
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 概要: 知识点1 构造函数与析构函数 知识点2 参数与返回值 知识点3 const 知识点4 函数重载(要与重写区分开) 知识点5 友元 先以C ...
- C++面向对象高级编程(七)point-like classes和function-like classes
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 1.pointer-like class 类设计成指针那样,可以当做指针来用,指针有两个常用操作符(*和->),所以我们必须重载这两个操作 ...
- C++面向对象高级编程(五)类与类之间的关系
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 本节主要介绍一下类与类之间的关系,也就是面向对象编程先介绍两个术语 Object Oriented Programming OOP面向对象编 ...
- C++面向对象高级编程(八)模板
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 这节课主要讲模板的使用,之前我们谈到过函数模板与类模板 (C++面向对象高级编程(四)基础篇)这里不再说明 1.成员模板 成员模板:参数为tem ...
- C++面向对象高级编程(六)转换函数与non-explicit one argument ctor
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 1.conversion function 转换函数 //1.转换函数 //conversion function //只要你认为合理 你可以任 ...
- C++面向对象高级编程(四)基础篇
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 一.Static 二.模板类和模板函数 三.namespace 一.Static 静态成员是“类级别”的,也就是它和类的地位等同,而普通成员是“ ...
- C++面向对象高级编程(三)基础篇
技术在于交流.沟通,转载请注明出处并保持作品的完整性. 概要 一.拷贝构造 二.拷贝赋值 三.重写操作符 四.生命周期 本节主要介绍 Big Three 即析构函数,拷贝构造函数,赋值拷贝函数,前面主 ...
- C++面向对象高级编程(上)-Geekband
头文件和类声明 一定要注意使用防卫式的头文件声明: #ifndef _CLASSHEAD_ #define _CLASSHEAD_ . . . . #endif 基于对象和面向对象 : 基于对象 单一 ...
随机推荐
- SqlServer PIVOT函数快速实现行转列,UNPIVOT实现列转行(转)
我们在写Sql语句的时候没经常会遇到将查询结果行转列,列转行的需求,拼接sql字符串,然后使用sp_executesql执行sql字符串是比较常规的一种做法.但是这样做实现起来非常复杂,而在SqlSe ...
- MySQL 储存过程-原理、语法、函数详细说明
Mysql储存过程是一组为了完成特定功能的SQL语句集,经过编译之后存储在数据库中,当需要使用该组SQL语句时用户只需要通过指定储存过程的名字并给定参数就可以调用执行它了,简而言之就是一组已经写好的命 ...
- [css]单/多行居中&字体设置
行高和字号 行高 CSS中,所有的行,都有行高.盒模型的padding,绝对不是直接作用在文字上的,而是作用在"行"上的. line-height: 40px; 文字,是在自己的行 ...
- django开发项目的部署nginx
Django 部署(Nginx) 本文主要讲解 nginx + uwsgi socket 的方式来部署 Django,比 Apache mod_wsgi 要复杂一些,但这是目前主流的方法. 1. 运行 ...
- 线性表:实现单链表和子类栈(Stack)及单向队列(Queue) [C++]
刚刚开始学习c++.之前c的内容掌握的也不多,基本只是一本概论课的程度,以前使用c的struct写过的链表.用python写过简单的数据结构,就试着把两者用c++写出来,也是对c++的class,以及 ...
- Struts2中struts.multipart.maxSize配置
今天使用Struts2的文件上传控件时,在struts.xml中,将处理上传的action中的fileUpload拦截器的maximumSize参数设置为5000000,上传了一个3M的文件后发现控制 ...
- HDU1059 二进制拆分优化多重背包
/*问你能不能将给出的资源平分成两半,那么我们就以一半为背包,运行多重背包模版 但是注意了,由于个数过大,直接运行会超时,所以要用二进制拆分每种的个数*/ #include<stdio.h> ...
- 337BRoutine Problem
/*给出你图片的长和宽的比例a:b 和摄像头的比例c:d,然后叫你求最后将图片放进摄像头 以后,剩余的面积比上摄像头的总面积,注意要化简为最简形式,而且摄像头要设置成至少一条边和图片相等 做法:先将两 ...
- SQL学习笔记之SQL查询练习题1
(网络搜集) 0x00 表名和字段 –1.学生表 Student(s_id,s_name,s_birth,s_sex) –学生编号,学生姓名, 出生年月,学生性别 –2.课程表 Course(c_id ...
- Ecplise项目转移到Android Studio,以及Genymotion模拟器介绍
一.移植android项目 今天简单分享一个从ecplise开发项目转移到Android Studio的方法,之前一直在ecplise上开发android项目,但是因为google现在主打Androi ...