admin管理员组文章数量:1122847
C++
1.1 语言基础
1.1.1 简述下C++语言的特点
参考回答:
C++在C语言基础上引入了面对对象的机制,同时也兼容C语言。
C++有三大特性(1)封装。(2)继承。(3)多态;
C++语言编写出的程序 结构清晰、易于扩充,程序可读性好。
C++生成的代码质量高,运行效率高,仅比汇编语言慢10%~20%****;
C++更加安全,增加了const常量、引用、四类cast转换(static_cast、dynamic_cast、
const_cast、reinterpret_cast)、智能指针、try—catch等等;
C++可复用性高,C++引入了模板的概念,后面在此基础上,实现了方便开发的标准模板库
STL(Standard Template Library)。
同时,C++是不断在发展的语言。C++后续版本更是发展了不少新特性,如C++11中引入了
nullptr、auto变量、Lambda匿名函数、右值引用、智能指针。
1.1.2 说说C语言和C++的区别
参考回答
C语言是C++的子集,C++可以很好兼容C语言。但是C++又有很多新特性,如引用、智能指针、
auto变量等。
C++是面对对象的编程语言;C语言是面对过程的编程语言。
C语言有一些不安全的语言特性,如指针使用的潜在危险**、强制转换的不确定性、内存泄露等。而
C++对此增加了不少新特性来改善安全性,如const常量、引用、cast转换、智能指针**、try—catch
等等;
C++可复用性高,C++引入了模板的概念,后面在此基础上,实现了方便开发的标准模板库STL。
C++的STL库相对于C语言的函数库更灵活、更通用。
1.1.3 说说 C++中 struct 和 class 的区别
参考回答
struct 一般用于描述一个数据结构集合,而 class 是对一个对象数据的封装;
struct 中默认的访问控制权限是 public 的,而 class 中默认的访问控制权限是 private 的,例如:
struct A{
int iNum;
// 默认访问控制权限是 public
}
class B{
int iNum;
// 默认访问控制权限是 private
}
在继承关系中,struct 默认是公有继承,而 class 是私有继承;
class 关键字可以用于定义模板参数,就像 typename,而 struct 不能用于定义模板参数,例如:
template<typename T, typename Y>
// 可以把typename 换成 class
int Func(const T& t, const Y& y) {
//TODO
}
答案解析
C++ 中的 struct 是对 C 中的 struct 进行了扩充,它们在声明时的区别如下:
image-20220710140722033
使用时的区别:C 中使用结构体需要加上 struct 关键字,或者对结构体使用 typedef 取别名,而
C++ 中可以省略 struct 关键字直接使用,例如:
struct Student{
int iAgeNum;
string strName;
}
typedef struct Student Student2;
//C中取别名
struct Student stu1; // C 中正常使用
Student2 stu2; // C 中通过取别名的使用
Student stu3; // C++ 中使用
1.1.4 说说include头文件的顺序以及双引号""和尖括号<>的区别
参考回答
区别:
(1)尖括号<>的头文件是系统文件,双引号""的头文件是自定义文件。
(2)编译器 预处理阶段 查找头文件的路径不一样。
查找路径:
(1)使用尖括号<>的头文件的查找路径:编译器设置的头文件路径–>系统变量。
(2)使用双引号""的头文件的查找路径:当前头文件目录–>编译器设置的头文件路径–>系统变
量。
1.1.5 说说C++结构体和C结构体的区别
参考回答
区别:
(1)C的结构体内**不允许有函数存在,C++**允许有内部成员函数,且允许该函数是虚函数。
(2)C的结构体对内部成员变量的访问权限只能是public,而C++允许public,protected,private三种。
(3)C语言的结构体是不可以继承的,C++的结构体是可以从其他的结构体或者类继承过来的。
(4)C 中使用结构体需要加上 struct 关键字,或者对结构体使用 typedef 取别名,而 C++ 中可以省略
struct 关键字直接使用。
1.1.6 导入C函数的关键字是什么,C++编译时和C有什么不同?
参考回答
关键字: 在C++中,导入C函数的关键字是extern,表达形式为extern “C”, extern "C"的主要作
用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分
代码按C语言的进行编译,而不是C++的。
编译区别: 由于C++支持函数重载,因此编译器编译函数的过程中会将函数的参数类型也加到编译
后的代码中,而不仅仅是函数名; 而C语言并不支持函数重载,因此编译C语言代码的函数时不会
带上函数的参数类型,一般只包括函数名。
答案解析
//extern示例
//在C++程序里边声明该函数,会指示编译器这部分代码按C语言的进行编译
extern “C” int strcmp(const char *s1, const char *s2);
//在C++程序里边声明该函数
extern “C”{
#include <string.h>//string.h里边包含了要调用的C函数的声明
}
//两种不同的语言,有着不同的编译规则,比如一个函数fun,可能C语言编译的时候为_fun,而C++则是__fun__
1.1.7 简述C++从代码到可执行二进制文件的过程
参考回答
C++和C语言类似,一个C++程序从源码到执行文件,有四个过程,预编译、编译、汇编、链接。
答案解析
预编译: 这个过程主要的处理操作如下:
(1) 将所有的#define删除,并且展开所有的宏定义
(2) 处理所有的条件预编译指令,如#if、#ifdef
(3) 处理#include预编译指令,将被包含的文件插入 到该预编译指令的位置。
(4) 过滤所有的注释
(5) 添加 行号和文件名标识。
2.编译:这个过程主要的处理操作如下:
(1) 词法分析: 将 源代码的字符序列 分割成一系列的记号。
(2) 语法分析: 对记号进行语法分析,产生语法树。
(3) 语义分析: 判断表达式是否有意义。
(4) 代码优化:
(5) 目标代码生成: 生成汇编代码。
(6) 目标代码优化:
3.汇编:这个过程主要是将汇编代码转变成机器可以执行的指令。
4.链接:将不同的源文件产生的目标文件进行链接,从而形成一个可以执行的程序。
链接分为静态链接和动态链接。
静态链接,是在链接的时候就已经把要调用的函数或者过程 链接到了生成的可执行文件中,就算你
在去把静态库删除也不会影响可执行程序的执行; 生成的静态链接库,Windows下以.lib为后缀,
Linux下以.a为后缀。
而动态链接,是在链接的时候 没有把调用的函数代码链接进去,而是在执行的过程中,再去找要链
接的函数, 生成的可执行文件中没有函数代码,只包含函数的重定位信息,所以当你删除动态库
时,可执行程序就不能运行。 生成的动态链接库,Windows下以.dll为后缀,Linux下以.so为后
缀。
1.1.8 说说 static关键字的作用
定义全局静态变量和局部静态变量:在变量前面加上static关键字。初始化的静态变量会在数据段
分配内存, 未初始化的静态变量会 在BSS段分配内存。直到程序结束,静态变量始终会维持前值。
只不过 全局静态变量 和局部静态变量的作用域不一样;
定义静态函数:在函数返回类型前加上static关键字,函数即被定义为静态函数。静态函数只能在
本源文件 中使用;
在变量类型前加上static关键字,变量即被定义为静态变量。静态变量只能在本源文件中使用
//示例
static int a;
static void func();
在c++中, static关键字 可以用于 定义类中的静态成员变量: 使用静态数据成员,它既可以被当成
全局变量那样去存储,但又被隐藏在类的内部。类中的static静态数据成员 拥有一块单独的存储
区,而不管创建了多少个该类的对象。所有这些对象的静态数据成员都共享这一块静态存储空间。
在c++中,static关键字可以用于定义类中的静态成员函数:与静态成员变量类似,类里面同样可
以定义静态成员函数。只需要在函数前加上关键字static即可。如 静态成员函数 也是类的一部分,
而不是对象的一部分。所有这些对象的静态数据成员都共享这一块静态存储空间。
答案解析
**
调用一个对象的 非静态成员函数时, 系统会把 该对象的起始地址赋给成员函数的this指针。而 静态成
员函数不属于任何一个对象,因此C++规定静态成员函数没有this指针(划重点,面试题常考)。既然
它没有指向 某一对象,也就无法对一个对象中的非静态成员进行访问。**
1.1.9 说说数组和指针的区别
参考回答
概念:
(1)数组:数组是用于储存多个相同类型数据的集合。 数组名是首元素的地址。
(2)指针:指针相当于一个变量,但是它和不同变量不一样,它存放的是其它变量在内存中的地
址。 指针名指向了内存的首地址。
区别:
(1) 赋值: 同类型指针变量可以相互赋值; 数组不行,只能一个一个元素的赋值或拷贝
(2) 存储方式:
数组:数组在内存中是连续存放的,开辟一块连续的内存空间。数组是根据数组的下进行访问
的,数组的存储空间,不是在静态区就是在栈上。
指针:指针很灵活,它可以指向任意类型的数据。指针的类型说明了它所指向地址空间的内
存。由于指针本身就是一个变量,再加上它所存放的也是变量,所以指针的存储空间不能确定。
(3) 求sizeof:
数组所占存储空间的内存大小: sizeof(数组名)/sizeof(数据类型)
在32位平台下,无论指针的类型是什么,sizeof(指针名)都是4, 在64位平台下,无论指针的
类型是什么,sizeof(指针名)都是8。
(4)初始化:
// 数组
int a[5] = { 0 };
char b[] = “Hello”; // 按字符串初始化,大小为6
char c[] = { ‘H’,‘e’,‘l’,‘l’,‘o’,‘\0’ };
int* arr = new int[10]; // 动态创建一维数组
// 指针
// 指向对象的指针
int* p = new int(0);
delete p;
// 指向数组的指针
int* p1 = new int[10];
delete[] p1;
// 指向类的指针:
string* p2 = new string;
delete p2;
// 按字符初始化// 指向指针的指针(二级指针)
int** pp = &p;
**pp = 10;
(5)指针操作:
数组名的指针操作
int a[3][4];
int (p)[4]; //该语句是定义一个数组指针,指向含4个元素的一维数组
p = a;
//将该二维数组的首地址赋给p,也就是a[0]或&a[0][0]
p++; //该语句执行过后,也就是p=p+1;p跨过行a[0][]指向了行a[1][]
//所以数组指针也称指向一维数组的指针,亦称行指针。
//访问数组中第i行j列的一个元素,有几种操作方式:
//(p[i]+j)、((p+i)+j)、((p+i))[j]、p[i][j]。其中,优先级:()>[]>。
//这几种操作方式都是合法的。
指针变量的数据操作:
char *str = “hello,douya!”;
str[2] = ‘a’;
*(str+2) = ‘b’;
//这两种操作方式都是合法的。
1.1.10 说说什么是函数指针,如何定义函数指针,有什么使用场景
参考回答
概念: 函数指针 就是指向函数的指针变量。每一个函数都有一个入口地址,该入口地址就是函数指
针所指向的地址。
定义形式如下:
int func(int a);
int (*f)(int a);
f = &func;
函数指针的应用场景:回调(callback)。
我们调用别人提供的 API函数(Application
Programming Interface,应用程序编程接口),称为Call; 如果别人的库里面调用我们的函数,就叫
Callback。
答案解析:
//以库函数qsort排序函数为例,它的原型如下:
void qsort(void base,//void类型,代表原始数组
size_t nmemb, //第二个是size_t类型,代表数据数量
size_t size, //第三个是size_t类型,代表单个数据占用空间大小
int(*compar)(const void *,const void *)//第四个参数是函数指针
);
//第四个参数告诉qsort,应该使用哪个函数来比较元素,即只要我们告诉qsort比较大小的规则,它就可以
帮我们对任意数据类型的数组进行排序。在库函数qsort调用我们自定义的比较函数,这就是回调的应用。
//示例int num[100];
int cmp_int(const void* _a , const void* _b){//参数格式固定
int* a = (int*)_a; //强制类型转换
int* b = (int*)_b;
return *a - *b;
}
qsort(num,100,sizeof(num[0]),cmp_int); //回调
1.1.11 说说静态变量什么时候初始化?
参考回答
对于C语言的全局和静态变量,初始化 发生在任何代码执行之前,属于 编译期 初始化。
而C++标准 规定:全局或静态 对象 当且 仅当 对象首次用到时才进行构造。
答案解析
作用域: C++里作用域可分为6种: 全局 ,局部, 类 ,语句, 命名空间 和 文件作用域。
静态全局变量 :全局作用域+文件作用域, 所以无法在 其他文件中使用。
静态局部变量 :局部作用域, 只被初始化一次,直到程序结束。
类静态成员变量: 类作用域。
所在空间:都在 静态存储区。因为静态变量 都在 静态存储区,所以下次调用函数的时候还是能取到
原来的值。
生命周期:静态全局变量、静态局部变量都在静态存储区,直到程序结束才会回收内存。类静态成
员变量在静态存储区,当超出类作用域时回收内存。
1.1.12 nullptr调用成员函数可以吗?为什么?
能。
原因:因为 在编译时 对象 就绑定了函数地址,和指针空不空没关系。
答案解析:
//给出实例
class animal{
public:
void sleep(){ cout << “animal sleep” << endl; }
void breathe(){ cout << “animal breathe haha” << endl; }
};
class fish :public animal{
public:
void breathe(){ cout << “fish bubble” << endl; }
};
int main(){
animal pAn=nullptr;
pAn->breathe(); // 输出:animal breathe haha
fish pFish = nullptr;
pFish->breathe(); // 输出:fish bubble return 0;
}
原因: 因为在编译时 对象就绑定了 函数地址,和 指针 空不空没关系。pAn->breathe();编译的时候,函数*
的地址就和指针pAn绑定了;调用breath(this), this就等于pAn。 由于函数中没有需要解引用this的地
方,所以函数运行不会出错,但是若用到this,因为this=nullptr,运行出错。
1.1.13 说说什么是野指针,怎么产生的,如何避免?
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)
产生原因:释放内存后 指针不及时置空(野指针),依然指向了该内存,那么可能出现非法访问的
错误。这些我们都要注意避免。
避免办法:
1)初始化置NULL
(2)申请内存后 判空
(3)指针释放后 置NULL
(4)使用智能指针
答案解析
产生原因: 释放内存后指针不及时置空(野指针),依然指向了该内存,那么可能出现非法访问的错
误。这些我们都要注意避免。如:
char *p = (char *)malloc(sizeof(char)*100);
strcpy(p, “Douya”);
free§;//p所指向的内存被释放,但是p所指的地址仍然不变
…
if (p != NULL){//没有起到防错作用
strcpy(p, “hello, Douya!”);//出错
}
避免办法:
(1)初始化置NULL
(2)申请内存后判空
(3)指针释放后置NULL
int *p = NULL; //初始化置NULL
p = (int *)malloc(sizeof(int)*n); //申请n个int内存空间
assert(p != NULL); //判空,防错设计
p = (int *) realloc(p, 25);//重新分配内存, p 所指向的内存块会被释放并分配一个新的内存地址
free§;
p = NULL; //释放后置空
int *p1 = NULL; //初始化置NULL
p1 = (int *)calloc(n, sizeof(int)); //申请n个int内存空间同时初始化为0
assert(p1 != NULL); //判空,防错设计free(p1);
p1 = NULL; //释放后置空
int *p2 = NULL; //初始化置NULL
p2 = new int[n]; //申请n个int内存空间
assert(p2 != NULL); //判空,防错设计
delete []p2;
p2 = nullptr; //释放后置空
1.1.15 说说内联函数和宏函数的区别
参考回答
区别:
宏定义不是函数,但是使用起来像函数。预处理器用复制宏代码的方式代替函数的调用,省去了函
数压栈退栈过程,提高了效率;而内联函数本质上是一个函数, 内联函数一般用于函数体的代码比
较简单的函数,不能包含复杂的控 制语句,while、switch,并且内联函数本身不能直接调用自
身。
宏函数是在预编译的时候 把所有的宏名 用 宏体来替换,简单的说就是字符串替换 ;而内联函数则
是在编译的时候进行代码插入,编译器 会在每处调用内联函数的地方直接把内联函数的内容展开,
这样可以省去函数的调用的开销,提高效率
宏定义是没有类型检查的,无论对还是错都是直接替换;而内联函数在编译的时候会进行类型的检
查,内联函数满足函数的性质,比如有返回值、参数列表等
答案解析
//宏定义示例
#define MAX(a, b) ((a)>(b)?(a):(b))
MAX(a, “Hello”); //错误地比较int和字符串,没有参数类型检查
//内联函数示例
#include <stdio.h>
inline int add(int a, int b) {
return (a + b);}
int main(void) {
int a;
a = add(1, 2);
printf(“a+b=%d\n”, a);
return 0;
}
//以上a = add(1, 2);处在编译时将被展开为:a = (a + b);
1、使用时的一些注意事项:
使用宏定义一定要注意错误情况的出现, 比如宏定义函数没有类型检查,可能传进来任意类型,从
而带来错误,如举例。还有就是括号的使用,宏在定义时要小心处理宏参数,一般用括号括起来,
否则容易出现二义性
inline函数一般用于比较小的,频繁调用的函数,这样可以减少函数调用带来的开销。只需要在函
数返回类型前加上关键字inline,即可将函数指定为inline函数。
同其它函数不同的是,最好将inline函数定义在头文件,而不仅仅是声明,因为编译器在处理
inline函数时,需要在调用点内联展开该函数,所以仅需要函数声明是不够的。
2、内联函数使用的条件:
内联是以代码膨胀(复制)为代价,仅仅省去了函数调用的开销,从而提高函数的执行效率。如果
执行函数体内代码的时间,相比于函数调用的开销较大,那么效率 的收获会很少。另一方面,每
一处内联函数的调用都要复制代码,将使程序的总代码量增大,消耗更多的内存空间。以下情况不
宜使用内联:
(1)如果函数体内的代码比较长,使用内联将导致内存消耗代价较高。
(2)如果函数体内出现循环,那么执行函数体内代码的时间要比函数调用的开销大。
内联不是什么时候都能展开的,一个好的编译器将会根据函数的定义体,自动地取消不符合要求的
内联。
1.1.16 说说运算符i++和++i的区别
参考回答
先看到实现代码:
#include <stdio.h>
int main(){
int i = 2;
int j = 2;
j += i++; //先赋值后加
printf(“i= %d, j= %d\n”,i, j); //i= 3, j= 4
i = 2;
j = 2;
j += ++i; //先加后赋值
printf(“i= %d, j= %d”,i, j); //i= 3, j= 5
}
赋值顺序不同:++ i 是先加后赋值;i ++ 是先赋值后加;++i和i++都是分两步完成的。
效率不同:后置++执行速度比前置的慢。
i++ 不能作为左值,而++i 可以:
int i = 0;
int* p1 = &(++i);//正确
// int* p2 = &(i++);//错误
++i = 1;//正确
// i++ = 1;//错误
两者都不是原子操作。
1.1.17 说说new和malloc的区别,各自底层实现原理。
参考回答:
new是操作符,而malloc是函数。
new在调用的时候 先分配内存,在调用构造函数,释放的时候 调用析构函数;而malloc没有构造
函数和析构函数。
malloc需要给定申请内存的大小,返回的指针需要强转; new会调用构造函数,不用指定内存的
大小,返回指针不用强转。
new可以被重载; malloc不行
new分配内存更直接和安全。
new发生 错误抛出异常,malloc 返回null
答案解析
malloc底层实现: 当开辟的空间小于 128K 时,调用 brk()函数;当开辟的空间大于 128K 时,调用
mmap()。
malloc采用的是内存池的管理方式,以减少内存碎片。先申请大块内存作为堆区,然后将
堆区分为多个内存块。当用户申请内存时,直接从堆区分配一块合适的空闲快。采用隐式链表将所有空
闲块,每一个空闲块记录了一个未分配的、连续的内存地址。
new底层实现: 关键字new在调用 构造函数的时候实际上进行了如下的几个步骤:
创建一个新的对象
将构造函数的作用域赋值给 这个新的对象(因此this指向了这个新的对象)
执行构造函数中的代码(为这个新对象添加属性)
返回新对象
1.1.18 说说const和define的区别。
参考回答
const用于定义常量;而define用于定义宏,而宏也可以用于定义常量。都用于常量定义时,它们的区
别有:
const生效于编译的阶段;define生效于预处理阶段。
const定义的常量,在C语言中是存储在内存中、需要额外的内存空间的; define定义的常量,运
行时是直接的操作数,并不会存放在内存中。
const定义的常量是 带类型的;define定义的常量不带类型。因此define定义的常量不利于类型检
查。
1.1.19 说说C++中函数指针和指针函数的区别。
参考回答
定义不同
指针函数 本质是 一个函数,其返回值为指针。
函数指针 本质是一个指针,其指向一个函数。
写法不同
指针函数:int fun(int x,int y);
函数指针:int (fun)(int x,int y);
用法不同
用法参考答案解析
答案解析
//指针函数示例
typedef struct _Data{
int a;
int b;
}Data;
//指针函数
Data* f(int a,int b){
Data * data = new Data;
//…
return data;
}
int main(){
//调用指针函数
Data * myData = f(4,5);
//Data * myData = static_cast<Data*>(f(4,5));
//…
}
//函数指针示例
int add(int x,int y){
return x+y;
}
//函数指针
int (*fun)(int x,int y);
//赋值
fun = add;
//调用
cout << "(*fun)(1,2) = " << (*fun)(1,2) ;
//输出结果
//(*fun)(1,2) =
3
1.1.20 说说const int *a, int const *a, const int a, int *const a, const int
*const a分别是什么,有什么特点。
参考回答
- const int a; //指的是a是一个常量,不允许修改。
- const int *a; //a指针所指向的内存里的值不变,即(*a)不变
- int const *a; //同const int *a;
- int *const a; //a指针所指向的内存地址不变,即a不变
- const int *const a; //都不变,即(*a)不变,a也不变
1.1.21 说说使用指针需要注意什么?
参考回答
定义指针时,先初始化为NULL。
用malloc或new申请内存之后,应该立即检查指针值是否为NULL。防止使用指针值为NULL的内
存。
不要忘记为数组和动态内存赋初值。防止将未被初始化的内存作为右值使用。
避免数字或指针的下标越界,特别要当心发生“多1”或者“少1”操作
动态内存的申请与释放必须配对,防止内存泄漏
用free或delete释放了内存之后,立即将指针设置为NULL,防止“野指针”
答案解析
(1)初始化置NULL
(2)申请内存后判空
(3)指针释放后置NULL
int *p = NULL; //初始化置NULL
p = (int *)malloc(sizeof(int)*n); //申请n个int内存空间
assert(p != NULL); //判空,防错设计
p = (int *) realloc(p, 25);//重新分配内存, p 所指向的内存块会被释放并分配一个新的内存地址
free§;
p = NULL; //释放后置空
int *p1 = NULL; //初始化置NULL
p1 = (int *)calloc(n, sizeof(int)); //申请n个int内存空间同时初始化为0
assert(p1 != NULL); //判空,防错设计
free(p1);
p1 = NULL; //释放后置空
int *p2 = NULL; //初始化置NULL
p2 = new int[n]; //申请n个int内存空间
assert(p2 != NULL); //判空,防错设计
delete []p2;
p2 = nullptr; //释放后置空
1.1.22 说说内联函数和函数的区别,内联函数的作用。
参考回答
内联函数比普通函数多了关键字inline
内联函数避免了函数调用的开销;普通函数有调用的开销
普通函数在被调用的时候,需要寻址(函数入口地址);内联函数 不需要寻址。
内联函数有一定的限制,内联函数体要求代码简单,不能包含复杂的结构控制语句;普通函数没有
这个要求。
内联函数的作用:内联函数在调用时,是将调用表达式用内联函数体来替换。避免函数调用的开销。
答案解析
在使用内联函数时,应注意如下几点:
在内联函数内不允许用循环语句和开关语句。
如果内联函数有这些语句,则编译将该函数视同普通函数那样产生函数调用代码,递归函数是不能
被用来做内联函数的。内联函数只适合于只有1~5行的小函数。对一个含有许多语句的大函数,
函数调用和返回的开销相对来说微不足道,所以也没有必要用内联函数实现。
内联函数的定义必须出现在内联函数第一次被调用之前。
1.1.23 简述C++有几种传值方式,之间的区别是什么?
参考回答
传参方式有这三种: 值传递、引用传递、指针传递
值传递:形参即使 在函数体内值发生变化,也不会影响实参的值;
引用传递:形参在 函数体内值发生变化,会影响实参的值;
指针传递: 在指针指向没有发生改变的前提下,形参在函数体内值发生变化,会影响实参的值;
答案解析:
值传递用于对象时,整个对象会拷贝一个副本,这样效率低;而引用传递用于对象时,不发生拷贝行
为,只是绑定对象,更高效;指针传递同理,但不如引用传递安全。
代码示例
//代码示例
#include
using namespace std;
void testfunc(int a, int *b, int &c){//形参a值发生了改变,但是没有影响实参i的值;但形参
b、c的值发生了改变,影响到了实参j、k的值
a += 1;
(*b) += 1;
c += 1;
printf(“a= %d, b= %d, c= %d\n”,a,*b,c);//a= 2, b= 2, c= 2
}
int main(){
int i = 1;
int a = 1;
int *j = &a;
int k = 1;
testfunc(i, j, k);
printf(“i= %d, j= %d, k= %d\n”,i,*j,k);//i= 1, j= 2, k= 2
return 0;
}
1.1.24 简述const(星号)和(星号)const的区别
参考回答
//const* 是常量指针,*const 是指针常量
int const *a; //a指针所指向的内存里的值不变,即(*a)不变
int *const a; //a指针所指向的内存地址不变,即a不变
1.2 C++内存
1.2.1 简述一下堆和栈的区别
参考回答
区别:
堆栈空间分配不同。栈由操作系统自动分配释放 ,存放函数的参数值,局部变量的值等;堆一般
由程序员分配释放。
堆栈缓存方式不同。栈使用的是一级缓存, 它们通常都是被调用时处于存储空间中,调用完毕立
即释放; 堆则是存放在二级缓存中,速度要慢些。
堆栈数据结构不同。 堆类似 数组结构;栈 类似 栈结构,先进后出。
1.2.2 简述C++的内存管理
参考回答
内存分配方式:
在C++中,内存分成5个区,他们分别是堆、栈、自由存储区、全局/静态存储区和常量存储区。
栈,在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元
自动被释放。
堆,就是那些由new分配的内存块,一般一个new就要对应一个delete。
自由存储区,就是那些由malloc等分配的内存块,和堆是十分相似的,不过是用free来结束自己的
生命。
全局/静态存储区,全局变量和静态变量被分配到同一块内存中
常量存储区,这是一块比较特殊的存储区,里面存放的是常量,不允许修改。
常见的内存错误及其对策:
(1)内存分配未成功,却使用了它。
(2)内存分配虽然成功,但是尚未初始化就引用它。
(3)内存分配成功并且已经初始化,但操作越过了内存的边界。
(4)忘记了释放内存,造成内存泄露。
(5)释放了内存却继续使用它。
对策:
(1)定义指针时,先初始化为NULL。
(2)用malloc或new申请内存之后,应该立即检查指针值是否为NULL。防止使用指针值为NULL
的内存。
(3)不要忘记为数组和动态内存赋初值。防止将未被初始化的内存作为右值使用。
(4)避免数字或指针的下标越界,特别要当心发生“多1”或者“少1”操作
(5)动态内存的申请与释放必须配对,防止内存泄漏
(6)用free或delete释放了内存之后,立即将指针设置为NULL,防止“野指针”
(7)使用智能指针。
内存泄露及解决办法:
什么是内存泄露?
简单地说就是申请了一块内存空间,使用完毕后没有释放掉。(1)new和malloc申请资源使用
后,没有用delete和free释放;(2)子类继承父类时,父类析构函数不是虚函数。(3)
Windows句柄资源使用后没有释放。
怎么检测?
第一:良好的编码习惯,使用了内存分配的函数,一旦使用完毕,要记得使用其相应的函数释放
掉。
第二:将分配的内存的指针以链表的形式自行管理,使用完毕之后从链表中删除,程序结束时可检
查改链表。
第三:使用智能指针。
第四:一些常见的工具插件,如ccmalloc、Dmalloc、Leaky、Valgrind等等。
1.2.3 malloc和局部变量分配在堆还是栈?
参考回答
malloc是在堆上分配内存,需要程序员自己回收内存;局部变量是在栈中分配内存,超过作用域就
自动回收。
1.2.4 程序有哪些section,分别的作用?程序启动的过程?怎么判断数据分配在栈
上还是堆上?
参考回答
image-20220711091130459
一个程序有哪些section:
如上图,从低地址到高地址,一个程序由代码段、数据段、BSS段、堆、共享区、栈 等组成。
数据段: 存放程序中已初始化的全局变量和静态变量的一块内存区域。
代码段: 存放 程序执行代码的一块内存区域。只读,代码段的头部还会包含一些只读的常数变量。
BSS 段: 存放程序中 未初始化的全局变量和静态变量的一块内存区域。
可执行程序在运行时又会多出两个区域:堆区和栈区。
堆区:动态申请内存用。堆从低地址向高地址增长。
栈区:存储局部变量、函数参数值。栈从高地址向低地址增长。是一块连续的空间。
最后还有一个共享区,位于堆和栈之间。
程序启动的过程:
操作系统 首先 创建相应的进程 并分配私有的进程空间 ,然后操作系统的加载器 负责把可执行文件的
数据段和代码段映射到 进程的虚拟内存空间中。
加载器 读入 **可执行程序的导入符号表,根据这些符号表可以查找出该可执行程序的所有依赖的动态链接库。
加载器 针对该程序的每一个动态链接库 调用 LoadLibrary
(1)查找对应的 动态库文件,加载器为该动态链接库确定一个合适的基地址。
(2)加载器 读取 该 动态链接库的导入符号表和导出符号表,比较 应用程序要求的导入符号是否匹
配 该库 的导出符号。
(3)针对 该库的导入符号表,查找对应的依赖的动态链接库,如有跳转,则跳到3
(4)调用该 动态链接库的初始化函数
初始化 应用程序的全局变量,对于 全局对象自动调用构造函数。
进入应用程序入口点函数开始执行。
怎么判断数据分配在栈上还是堆上:
首先局部变量分配在栈上;而通过malloc和new申请的空间是在堆上。
1.2.5 初始化为0的全局变量在bss还是data
参考回答
BSS段通常 是指用来存放程序中 未初始化的或者初始化为0的全局变量和静态变量的一块内存区域。特
点是可读写的,在程序执行之前BSS段会自动清0。
1.2.6 什么是内存泄露,内存泄露怎么检测?
参考回答
什么是内存泄露?
简单地说就是申请了一块内存空间,使用完毕后没有释放掉。(1)new和malloc申请资源使用后,没
有用delete和free释放;(2)子类继承父类时,父类析构函数不是虚函数。(3)Windows句柄资源使
用后没有释放。
怎么检测?
第一:良好的编码习惯,使用了内存分配的函数,一旦使用完毕,要记得使用其相应的函数释放掉。
第二:将分配的内存的指针以链表的形式自行管理,使用完毕之后从链表中删除,程序结束时可检查改
链表。
第三:使用智能指针。
第四:一些常见的工具插件,如ccmalloc、Dmalloc、Leaky、Valgrind等等。
1.2.7 请简述一下atomoic内存顺序。
参考回答
有 六个内存顺序选项可应用于对原子类型的操作:
memory_order_relaxed: 在原子类型上的操作以自由序列执行,没有任何同步关系,仅对此操作
要求原子性。
memory_order_consume: memory_order_consume只会对其标识的对象保证 该对象存储 先行
于 那些需要加载该对象的操作。
memory_order_acquire: 使用memory_order_acquire的原子操作,当前线程的读写操作 都不能
重排到此操作之前。
memory_order_release: 使用memory_order_release的原子操作,当前线程的读写操作 都不能
重排到此 操作之后。
memory_order_acq_rel: memory_order_acq_rel在此内存顺序的读-改-写操作 既是获得加载又
是释放操作。没有操作 能够从此操作之后被 重排到此操作之前,也没有操作 能够 从此 操作之前被重
排到此操作之后。
memory_order_seq_cst :memory_order_seq_cst比std::memory_order_acq_rel更为严格。
memory_order_seq_cst不仅是一个"获取释放"内存顺序,它还会对所有拥有此标签的内存操作建
立一个单独全序。
除非你为特定的操作指定一个顺序选项,否则内存顺序选项 对于所有原子类型默认都是
memory_order_seq_cst。
1.2.8 内存模型,堆栈,常量区。
参考回答
image-20220711092648944
内存模型(内存布局):
如上图,从低地址到高地址,一个程序由代码段、数据段、BSS段、堆、共享区、栈等组成。
数据段: **** 存放程序中已初始化的全局变量和静态变量的一块内存区域。
代码段 : 存放程序执行代码的一块内存区域。只读,代码段的头部还会包含一些只读的常数变量。
BSS 段: 存放程序中未初始化的全局变量和静态变量的一块内存区域。
可执行程序在运行时又会多出两个区域:堆区和栈区。
堆区:动态申请内存用。堆从低地址向高地址增长。
栈区:存储局部变量、函数参数值。栈从高地址向低地址增长。是一块连续的空间。
最后还有一个共享区,位于堆和栈之间。
堆 heap :由new分配的内存块,其释放由程序员控制(一个new对应一个delete)
栈 stack :是那些编译器在需要时分配,在不需要时自动清除的存储区。存放局部变量、函数参数。
常量存储区 :存放常量,不允许修改。
1.2.9 简述C++中内存对齐的使用场景
参考回答
内存对齐应用于三种数据类型中:struct/class/union
struct/class/union内存对齐原则有四个:
数据成员对齐规则: 结构(struct)或联合(union)的数据成员, 第一个数据成员放在offset为0的地
方,以后 每个数据成员存储的起始位置 要从该成员大小 或者 成员的子成员大小 的整数倍开始。
结构体作为成员:如果 一个结构里有某些结构体成员,则结构体成员 要从其内部"最宽基本类型成
员"的整数倍地址开始存储。(struct a里存有struct b,b里有char,int ,double等元素,那b应该从8的
整数倍开始存储)。
收尾工作:结构体的总大小,也就是sizeof的结果,必须是其内部最大成员的"最宽基本类型成员"的
整数倍。不足的要补齐。(基本类型不包括struct/class/uinon)。
sizeof(union),以结构里面size最大元素为union的size,因为在某一时刻,union只有一个成员真
正存储于该地址。
答案解析
- 什么是内存对齐?
那么什么是字节对齐?在C语言中,结构体是一种复合数据类型,其构成元素既可以是基本数据类
型(如int、long、float等)的变量,也可以是一些复合数据类型(如数组、结构体、联合体等)
的数据单元。在结构体中,编译器为结构体的每个成员按其自然边界(alignment)分配空间。
各个成员 按照它们被声明的顺序在内存中顺序存储, 第一个成员的地址和整个结构体的地址相同。
为了使CPU能够对变量进行快速的访问,变量的起始地址应该具有某些特性,即所谓的“对齐”,比
如4字节的int型,其起始地址应该位于4字节的边界上,即起始地址能够被4整除,也即“对齐”跟数
据在内存中的位置有关。如果一个变量的内存地址正好位于它长度的整数倍,他就被称做自然对
齐。
比如在32位cpu下,假设一个整型变量的地址为0x00000004(为4的倍数),那它就是自然对齐的,
而如果其地址为0x00000002(非4的倍数)则是非对齐的。现代计算机中内存空间都是按照byte
划分的,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定
类型变量的时候经常在特定的内存地址访问,这就需要各种类型数据按照一定的规则在空间上排
列,而不是顺序的一个接一个的排放,这就是对齐。
- 为什么要字节对齐?
需要字节对齐的根本原因在于CPU访问数据的效率问题。假设上面整型变量的地址不是自然对齐,
比如为0x00000002,则CPU如果取它的值的话需要访问两次内存,第一次取从0x00000002-
0x00000003的一个short,第二次取从0x00000004-0x00000005的一个short然后组合得到所要
的数据, 如果变量在0x00000003地址上的话则要访问三次内存,第一次为char,第二次为
short,第三次为char,然后组合得到整型数据。
而如果变量在自然对齐位置上,则只要一次就可以取出数据。一些系统对对齐要求非常严格,比如
sparc系统,如果取未对齐的数据会发生错误,而在x86上就不会出现错误,只是效率下降。
各个硬件平台对存储空间的处理上有很大的不同。一些平台对某些特定类型的数据只能从某些特定
地址开始存取。比如有些平台每次读都是从偶地址开始,如果一个int型(假设为32位系统)如果
存放在偶地址开始的地方,那么一个读周期就可以读出这32bit,而如果存放在奇地址开始的地
方,就需要2个读周期,并对两次读出的结果的高低字节进行拼凑才能得到该32bit数据。显然在读
取效率上下降很多。
字节对齐实例
union example {
int a[5];
char b;
double c;
};
int result = sizeof(example);
/*如果以最长20字节为准,内部double占8字节,这段内存的地址0x00000020并不是double的整数
倍,只有当最小为0x00000024时可以满足整除double(8Byte)同时又可以容纳int a[5]的大小,
所以正确的结果应该是result=24
/
struct example {
int a[5];
char b;
double c;
}test_struct;
int result = sizeof(test_struct);
/
如果我们不考虑字节对齐,那么内存地址0x0021不是double(8Byte)的整数倍,所以需要字节对
齐,那么此时满足是double(8Byte)的整数倍的最小整数是0x0024,说明此时char b对齐int扩充
了三个字节。所以最后的结果是result=32
/
struct example {
char b;
double c;
int a;
}test_struct;
int result = sizeof(test_struct);
/
字节对齐除了内存起始地址要是数据类型的整数倍以外,还要满足一个条件,那就是占用的内存空间大
小需要是结构体中占用最大内存空间的类型的整数倍,所以20不是double(8Byte)的整数倍,我们还
要扩充四个字节,最后的结果是result=24
*/
1.3 面向对象
1.3.1 简述一下什么是面向对象
参考回答:
面向对象是一种编程思想,把一切东西看成是一个个 对象,比如人、耳机、鼠标、水杯等,他们各
自都有属性,比如:耳机是白色的,鼠标是黑色的,水杯是圆柱形的等等,把这些对象拥有的属性
变量和操作 这些属性变量的函数打包成一个类来表示
面向过程和面向对象的区别
面向过程:根据业务逻辑从上到下写代码
面向对象:将数据与函数绑定到一起,进行封装,这样能够更快速的开发程序,减少了重复代码的
重写过程
1.3.2 简述一下面向对象的三大特征
参考回答
面向对象的三大特征是封装、继承、多态。
封装: 将数据和操作数据的方法 进行有机结合, 隐藏对象的属性和实现细节,仅对外 公开接口来和
对象进行 交互。封装本质上是一种管理: 我们如何管理兵马俑呢?比如如果什么都不管,兵马俑
就被随意破坏了。那么我们首先建了一座房子把兵马俑给封装起来。但是我们目的 全封装起来,不
让别人看。所以我们开放了售票通 道,可以买票突破封装在合理的监管机制下进去参观。
类也是一样,不想给别人看到的,我们使用protected/private把成员封装起来。开放一些共有的成员函
数对成员合理的访问。所以封装本质是一种管理
继承:可以使用现有类的所有功能,并在无需重新编写原来的类的情况下对这些功能进行扩展。
三种继承方式
image-20220711095601102
多态: 用 父类型别的指针指向其子类的实例, 然后通过父类的指针调用实际子类的成员函数。实现
多态,有二种方式,重写,重载。
1.3.3 简述一下 C++ 的重载和重写,以及它们的区别
参考回答:
重写
是指派生类中存在重新定义的函数。其函数名,参数列表,返回值类型,所有都必须同基类中被重
写的函数一致。只有函数体不同(花括号内), 派生类对象调用时会调用派生类的重写函数,不会
调用被重写函数。重写的基类中被重写的函数必须有virtual修饰。
示例如下:
#include<bits/stdc++.h>
using namespace std;
class A
{
public:
virtual void fun()
{
cout << “A”;
}
};
class B :public A
{
public:
virtual void fun()
{
cout << “B”;
}
};
int main(void)
{
A* a = new B();
a->fun();//输出B,A类中的fun在B类中重写
}
重载
我们在平时写代码中会用到几个函数但是他们的实现功能相同,但是有些细节却不同。例如:交换
两个数的值其中包括(int, float,char,double)这些个类型。在C语言中我们是利用不同的函数名来
加以区分。这样的代码不美观而且给程序猿也带来了很多的不便。于是在C++中人们提出了用一个
函数名定义多个函数,也就是所谓的函数重载。函数重载是指同一可访问区内被声明的几个具有不
同参数列(参数的类型,个数,顺序不同)的同名函数,根据参数列表确定调用哪个函数,重载不关心函数返回类型。**
#include<bits/stdc++.h>
using namespace std;
class A
{
void fun() {};
void fun(int i) {};
void fun(int i, int j) {};
void fun1(int i,int j){};
};
1.3.4 说说 C++ 的重载和重写是如何实现的
参考答案
C++利用命名倾轧(name mangling)技术,来改名函数名, 区分参数不同的同名函数。命名倾轧是在编译阶段完成的。
C++定义同名重载函数:
#include
using namespace std;
int func(int a,double b)
{
return ((a)+(b));
}
int func(double a,float b)
{
return ((a)+(b));
}
int func(float a,int b)
{
return ((a)+(b));
}
int main()
{
return 0;
}
image-20220711100217706
由上图可得,d代表double,f代表float,i代表int,加上参数首字母以区分同名函数。
在基类的函数前加上virtual关键字,在派生类中重写该函数,运行时将会根据对象的实际类型来调
用相应的函数。如果对象类型是派生类,就调用派生类的函数; 如果对象类型是基类,就调用基类
的函数。
用virtual关键字申明的函数叫做虚函数,虚函数肯定是类的成员函数。
存在虚函数的类都有一个一维的虚函数表叫做虚表,类的对象有一个指向虚表开始的虚指
针。虚表是和类对应的,虚表指针 是和 对象对应的。
多态性是一个接口多种实现,是面向对象的核心,分为类的多态性和函数的多态性。
重写用虚函数来实现,结合动态绑定。
纯虚函数是虚函数再加上 = 0。
抽象类是 指包括至少一个纯虚函数的类。
纯虚函数 :virtual void fun()=0。即抽象类必须在子类实现这个函数,即先有名称,没有内容,在
派生类实现内容。
1.3.5 说说 C 语言如何实现 C++ 语言中的重载
参考答案
c语言中 不允许有同名函数,因为 编译时函数命名 是一样的,不像c++会添加参数类型和返回类型作为函
数编译后的名称,进而实现重载。如果要用c语言显现函数重载,可通过以下方式来实现:
使用 函数指针 来实现,重载的函数不能使用同名称,只是类似的实现了函数重载功能
重载函数 使用可变参数,方式如打开文件open函数
gcc有内置函数,程序使用编译函数可以实现函数重载
#include<stdio.h>
void func_int(void * a)
{
printf(“%d\n”,(int)a); //输出int类型,注意 void * 转化为int
}
void func_double(void * b)
{
printf(“%.2f\n”,(double)b);
}
typedef void (*ptr)(void *); //typedef申明一个函数指针void c_func(ptr p,void *param)
{
p(param); //调用对应函数
}
int main()
{
int a = 23;
double b = 23.23;
c_func(func_int,&a);
c_func(func_double,&b);
return 0;
}
1.3.6 说说构造函数有几种,分别什么作用
参考答案
C++中的构造函数可以分为4类:默认构造函数、初始化构造函数、拷贝构造函数、移动构造函数。
默认构造函数和初始化构造函数。 在定义类的对象的时候,完成对象的初始化工作。
class Student
{
public:
//默认构造函数
Student()
{
num=1001;
age=18;
}
//初始化构造函数
Student(int n,int a):num(n),age(a){}
private:
int num;
int age;
};
int main()
{
//用默认构造函数初始化对象S1
Student s1;
//用初始化构造函数初始化对象S2
Student s2(1002,18);
return 0;
}
有了有参的构造了,编译器就不提供默认的构造函数。
拷贝构造函数
#include “stdafx.h”
#include “iostream.h”
class Test
{
int i; int *p;
public:
Test(int ai,int value)
{
i = ai;
p = new int(value);
}
~Test()
{
delete p;
}
Test(const Test& t)
{
this->i = t.i;
this->p = new int(*t.p);
}
};
//复制构造函数用于复制本类的对象
int main(int argc, char* argv[])
{
Test t1(1,2);
Test t2(t1);//将对象t1复制给t2。注意复制和赋值的概念不同
return 0;
}
赋值构造函数默认实现的是值拷贝(浅拷贝)。
移动构造函数。用于将其他类型的变量,隐式转换为本类对象。下面的转换构造函数,将int类型
的r转换为Student类型的对象,对象的age为r,num为1004.
Student(int r)
{
int num=1004;
int age= r;
}
1.3.7 只定义析构函数,会自动生成哪些构造函数
参考答案
只定义了析构函数,编译器将自动为我们生成 拷贝构造函数和默认构造函数。
默认构造函数和初始化构造函数。 在定义类的对象的时候,完成对象的初始化工作。
class Student
{
public:
//默认构造函数
Student()
{
num=1001;
age=18;
}
//初始化构造函数
Student(int n,int a):num(n),age(a){}
private:int num;
int age;
};
int main()
{
//用默认构造函数初始化对象S1
Student s1;
//用初始化构造函数初始化对象S2
Student s2(1002,18);
return 0;
}
有了有参的构造了,编译器就不提供默认的构造函数。
拷贝构造函数
#include “stdafx.h”
#include “iostream.h”
class Test
{
int i;
int *p;
public:
Test(int ai,int value)
{
i = ai;
p = new int(value);
}
~Test()
{
delete p;
}
Test(const Test& t)
{
this->i = t.i;
this->p = new int(*t.p);
}
};
int main(int argc, char* argv[])
{
Test t1(1,2);
Test t2(t1);//将对象t1复制给t2。注意复制和赋值的概念不同。
return 0;
}
赋值构造函数默认实现的是值拷贝(浅拷贝)。
答案解析
示例如下:
class HasPtr
{
public:
HasPtr(const string& s = string()) :ps(new string(s)), i(0) {}
~HasPtr() { delete ps; }
private:
string * ps;
int i;
}
如果类外面有这样一个函数:
HasPtr f(HasPtr hp)
{
HasPtr ret = hp;
///… 其他操作
return ret;
}
当函数执行完了之后,将会调用hp和ret的析构函数,将hp和ret的成员ps给delete掉,但是由于ret和
hp指向了同一个对象,因此该对象的ps成员被delete了两次,这样产生一个未定义的错误,所以说,如
果一个类定义了析构函数,那么它要定义自己的拷贝构造函数和默认构造函数。
1.3.8 说说一个类,默认会生成哪些函数
定义一个空类
class Empty
{
};
默认会生成以下几个函数
无参的构造函数
在定义类的对象的时候,完成对象的初始化工作。
Empty()
{
}
拷贝构造函数
拷贝构造函数用于复制本类的对象
Empty(const Empty& copy)
{
}
赋值运算符
Empty& operator = (const Empty& copy)
{
}
析构函数(非虚)
~Empty()
{
}
1.3.9 说说 C++ 类对象的初始化顺序,有多重继承情况下的顺序
参考答案:
创建派生类的对象, 基类的构造函数 优先被调用(也优先于派生类里的成员类);
如果类里面有 成员类, 成员类的构造函数 优先被调用; (也优先于该类本身的构造函数)
基类构造函数如果有多个基类, 则构造函数的调用顺序 是某类 在类派生表中 出现的顺序 而不是它们
在成员初始化表中的顺序;
成员类对象 构造函数 如果有多个成员 类对象,则构造函数的调用顺序 是 对象在类中被声明的顺序而
不是它们出现在成员初始化表中的顺序;
派生类构造函数,作为一般规则派生类构造函数 应该不能直接向一个基类数据成员赋值 而是把值传
递给 适当的基类构造函数,否则两个类的实现变成紧耦合的(tightly coupled)将更加难于正确地
修改或扩展基类的实现。(基类设计者的责任是提供一组适当的基类构造函数)
综上可以得出,初始化顺序:
父类构造函数–>成员类对象构造函数–>自身构造函数
其中 成员变量的初始化与声明顺序有关,构造函数的调用顺序是 类派生列表中的顺序。
析构顺序和构造顺序相反。
1.3.10 简述下向上转型和向下转型
子类转换为父类: 向上转型, 使用dynamic_cast<type_id>(expression), 这种转换相对来说比较
安全不会有数据的丢失;
父类转换为子类: 向下转型,可以使用强制转换,这种转换时不安全的,会导致数据的丢失, 原因
是 父类的指针或者引用的内存中 可能不包含 子类的成员的内存。
1.3.11 简述下深拷贝和浅拷贝,如何实现深拷贝
浅拷贝: 又称值拷贝,将源对象的值 拷贝到 目标对象中去,本质上来说 源对象和目标对象 共用一份
实体,只是所引用的变量名不同,地址其实还是相同的。举个简单的例子,你的小名叫西西,大名
叫冬冬, 当别人叫你西西或者冬冬的时候你都会答应,这两个名字虽然不相同,但是都指的是你。
2.深拷贝,拷贝的时候先开辟出和源对象大小一样的空间,然后将源对象里的内容拷贝到目标对象中
去,这样两个指针就指向了不同的内存位置。并且里面的内容是一样的,这样不但达到了我们想要
的目的,还不会出现问题,两个指针先后去调用析构函数,分别释放自己所指向的位置。即为每次
增加一个指针,便申请一块新的内存,并让这个指针指向新的内存,深拷贝情况下,不会出现重复
释放同一块内存的错误。
深拷贝的实现: 深拷贝的拷贝构造函数和赋值运算符的重载传统实现:
STRING( const STRING& s )
{
//_str = s._str;
_str = new char[strlen(s._str) + 1];
strcpy_s( _str, strlen(s._str) + 1, s._str );
}
STRING& operator=(const STRING& s)
{
if (this != &s)
{
//this->_str = s._str;
delete[] _str;
this->_str = new char[strlen(s._str) + 1];
strcpy_s(this->_str, strlen(s._str) + 1, s._str);
}
return *this;
}
这里的拷贝构造函数我们很容易理解,先开辟出和源对象一样大的内存区域,然后将需要拷贝的数
据复制到目标拷贝对象 , 那么这里的赋值运算符的重载是怎么样做的呢?
image-20220711105807280
这种方法解决了我们的指针悬挂问题,通过不断的开空间 让不同的指针指向不同的内存,以防止同
一块内存被释放两次的问题。
1.3.12 简述一下 C++ 中的多态
由于 派生类重写 基类方法,然后用 基类引用指向派生类对象,调用方法时候会进行动态绑定,这就是多
态。 多态分为静态多态和动态多态:
静态多态:编译器在编译期间完成的,编译器会根据实参类型来推断该调用哪个函数,如果有对应
的函数,就调用, 没有则在编译时报错。
include
using namespace std;
int Add(int a,int b)//1
{
return a+b;
}
char Add(char a,char b)//2
{
return a+b;
}
int main()
{
cout<<Add(666,888)<<endl;//1
cout<<Add(‘1’,‘2’);//2
return 0;
}
显然,第一条语句会调用函数1,而第二条语句会调用函数2,这绝不是因为函数的声明顺序,不
信你可以将顺序调过来试试。
动态多态:其实要实现动态多态,需要几个条件——即动态绑定条件:
虚函数。基类中必须有虚函数,在派生类中必须重写虚函数。
通过基类类型的指针或引用来调用虚函数。
说到这,得插播一条概念:重写——也就是基类中有一个虚函数,而在派生类中也要重写一个原型
(返回值、名字、参数)都相同的虚函数。不过协变例外。协变是重写的特例,基类中 返回值 是基
类类型的引用或指针,在派生类中,返回值为 派生类类型的引用或指针。
//协变测试函数
#include
using namespace std;
class Base
{
public:
virtual Base* FunTest()
{
cout << “victory” << endl;
return this;
}
};
class Derived :public Base
{
public:
virtual Derived* FunTest()
{
cout << “yeah” << endl;
return this;
};
int main()
{
Base b;
Derived d;
b.FunTest();
d.FunTest();
return 0;
}
1.3.13 说说为什么要虚析构,为什么不能虚构造
虚析构: 将可能会被继承的父类的析构函数设置为虚函数,可以保证当我们new一个子类,然后使
用基类指针指向该子类对象,释放基类指针时可以释放掉子类的空间,防止内存泄漏。如果基类的
析构函数不是虚函数,在特定情况下会导致派生来 无法被析构。
用派生类类型指针绑定 派生类实例, 析构的时候,不管基类析构函数是不是虚函数,都会正
常析构
用基类类型指针绑定派 生类实例, 析构的时候,如果基类析构函数不是虚函数,则只会析构
基类, 不会析构派生类对象,从而造成内存泄漏。
为什么会出现这种现象呢,个人认为析构的时候如果没有虚函数的动态绑定功能,就只根据指针的类型来进行的,而不是根据指针绑
定的对象来进行,所以只是调用了基类的析构函数;如果基类的析构函数是虚函数,则析构的时候就要根据指针绑定的对象来调用对应的析构函数了。
C++默认的析构函数不是虚函数是因为虚函数需要额外的虚函数表和虚表指针,占用额外的内存。
而对于不会被继承的类来说,其析构函数如果是虚函数,就会浪费内存。因此C++默认的析构函数
不是虚函数,而是只有当需要当作父类时,设置为虚函数。
不能虚构造
从存储空间角度: 虚函数 对应一个vtale,这个表的地址 是存储在对象的内存空间的。 如果将构
造函数设置为虚函数,就需要到vtable 中调用, 可是对象还没有实例化,没有内存空间分
配,如何调用。(悖论)
从使用角度: 虚函数主要用于在信息不全的情况下, 能使重载的函数得到对应的调用。构造
函数本身就是要初始化实例,那使用虚函数也没有实际意义呀。所以构造函数 没有必要是虚
函数。虚函数的作用 在于通过父类的指针或者引用来调用它的 时候能够变成调用子类的那个
成员函数。而构造函数是在创建对象时自动调用的, 不可能通过父类的指针或者引用去调
用,因此也就规定构造函数不能是虚函数。
从实现上看, vbtl 在构造函数调用后才建立,因而构造函数不可能成为虚函数。从实际含义
上看,在调用构造函数时还不能确定对象的真实类型(因为子类会调父类的构造函数);而
且构造函数的作用是 提供初始化,在对象生命期只执行一次, 不是对象的动态行为,也没有
太大的必要成为虚函数。
1.3.14 说说模板类是在什么时候实现的
模板实例化: 模板的实例化 分为显示实例化和隐式实例化, 前者是研发人员明确的告诉模板应该使
用什么样的类型去生成具体的类或函数, 后者是在编译的过程中由编译器来决定使用什么类型来实
例化 一个模板不管是显示实例化或隐式实例化,最终生成的类或函数完全是按照模板的定义来实现
的
模板具体化: 当模板使用某种类型类型实例化后生成的类或函数不能满足需要时,可以考虑对模板
进行具体化。具体化时可以修改原模板的定义,当使用该类型时,按照具体化后的定义实现,具体
化相当于对某种类型进行特殊处理。
代码示例:
#include
using namespace std;
// #1 模板定义
template
struct TemplateStruct
{
TemplateStruct()
{
cout << sizeof(T) << endl;
}
};
// #2 模板显示实例化
template struct TemplateStruct;
// #3 模板具体化
template<> struct TemplateStruct
{
TemplateStruct() {
cout << “–8–” << endl;
}
};
int main()
{
TemplateStruct intStruct;
TemplateStruct doubleStruct;
// #4 模板隐式实例化
TemplateStruct llStruct;
}
运行结果
4
–8–
1
1.3.15 说说类继承时,派生类对不同关键字修饰的基类方法的访问权限
类中的成员可以分为三种类型,分别为public成员、protected成员、public成员。类中可以直接访问自
己类的public、protected、private成员,但类对象只能访问自己类的public成员。
public继承:派生类可以访问基类的public、protected成员,不可以访问基类的private成员;
派生类对象可以访问基类的public成员,不可以访问基类的protected、private成员。
protected继承:派生类可以访问基类的public、protected成员,不可以访问基类的private成员;
派生类对象不可以访问基类的public、protected、private成员。
private继承:派生类可以访问基类的public、protected成员,不可以访问基类的private成员;
派生类对象不可以访问基类的public、protected、private成员。
1.3.16 简述一下移动构造函数,什么库用到了这个函数?
C++11中新增了移动构造函数。与拷贝类似,移动也使用一个对象的值设置另一个对象的值。但是,又
与拷贝不同的是,移动实现的是对象值真实的转移(源对象到目的对象): 源对象将丢失其内容,其内
容将被目的对象占有。
移动操作的发生的时候,是 当移动值的对 象是未命名的对象的时候。这里未命名
的对象就是那些临时变量,甚至都不会有名称。典型的未命名对象就是函数的返回值或者类型转换的对
象。使用临时对象的值初始化另一个对象值,不会要求对对象的复制:因为临时对象不会有其它使用,
因而,它的值可以被移动到目的对象。做到这些,就要使用移动构造函数和移动赋值:当使用一个临时
变量对对象进行构造初始化的时候,调用移动构造函数。类似的,使用未命名的变量的值赋给一个对象
时,调用移动赋值操作。
移动操作的概念 对 对象管理它们使用的存储空间很有用的,诸如对象使用new和delete分配内存的时
候。在这类对象中,拷贝和移动是不同的操作:从A拷贝到B意味着,B分配了新内存,A的整个内容被
拷贝到为B分配的新内存上。
而 从A移动到B意味着 分配给A的内存转移给了B,没有分配新的内存,它仅仅包含简单地拷贝指针。
看下面的例子:
// 移动构造函数和赋值
#include
#include
using namespace std;
class Example6 {
string* ptr;
public:
Example6 (const string& str) : ptr(new string(str)) {}
~Example6 () {delete ptr;}
// 移动构造函数,参数x不能是const Pointer&& x,
// 因为要改变x的成员数据的值;
// C++98不支持,C++0x(C++11)支持
Example6 (Example6&& x) : ptr(x.ptr)
{
x.ptr = nullptr;
}
// move assignment
Example6& operator= (Example6&& x)
{
delete ptr;
ptr = x.ptr;
x.ptr=nullptr;
return *this;
}
// access content:
const string& content() const {return *ptr;}
// addition:
Example6 operator+(const Example6& rhs)
{
return Example6(content()+rhs.content());
}
};
int main () {
Example6 foo(“Exam”); // 构造函数
// Example6 bar = Example6(“ple”); // 拷贝构造函数
Example6 bar(move(foo)); // 移动构造函数
// 调用move之后,foo变为一个右值引用变量,
// 此时,foo所指向的字符串已经被"掏空",
// 所以此时不能再调用foo
bar = bar+ bar; // 移动赋值,在这儿"="号右边的加法操作,
// 产生一个临时值,即一个右值
// 所以此时调用移动赋值语句
cout << "foo’s content: " << foo.content() << ‘\n’;
return 0;
}
执行结果:
foo’s content: Example
1.3.17 请你回答一下 C++ 类内可以定义 引用数据成员吗?
c++ 类内 可以定义引用成员变量,但要遵循以下三个规则:
不能 用默认构造函数初始化,必须提供构造函数来初始化引用成员变量。否则会造成引用未初始化
错误。
构造函数的形参 也必须是引用类型。
不能在构造函数里初始化,必须在初始化列表中进行初始化。
1.3.18 构造函数为什么不能被声明为虚函数?
从存储空间角度: 虚函数对应一个vtale,这个表的地址是存储在对象的内存空间的。如果将构造函
数设置为虚函数,就需要到vtable 中调用,可是对象还没有实例化,没有内存空间分配,如何调
用。(悖论)
从使用角度 :虚函数主要用于在信息不全的情况下,能使重载的函数得到对应的调用。构造函数本
身就是要初始化实例,那使用虚函数也没有实际意义呀。所以构造函数没有必要是虚函数。虚函数
的作用在于通过父类的指针或者引用来调用它的时候能够变成调用子类的那个成员函数。而构造函
数是在创建对象时自动调用的,不可能通过父类的指针或者引用去调用,因此也就规定构造函数不
能是虚函数。
从实现上看, vbtl 在构造函数调用后才建立,因而构造函数不可能成为虚函数。从实际含义上看,
在调用构造函数时还不能确定对象的真实类型(因为子类会调父类的构造函数);而且构造函数的
作用是提供初始化,在对象生命期只执行一次,不是对象的动态行为,也没有太大的必要成为虚函
数。
1.3.19 简述一下什么是常函数,有什么作用
类的成员函数后面加 const, 表明这个函数不会对这个类对象的数据成员(准确地说是非静态数据成
员)作任何改变。 在设计类的时候,一个原则 就是对于不改变数据成员的成员函数都要在后面加
const,而对于改变数据成员的成员函数不能加 const。所以 const 关键字对成员函数的行为作了更明
确的限定: 有 const 修饰的成员函数 (指 const 放在函数参数表的后面, 而不是在函数前面或者参数表
内),只能读取数据成员,不能改变数据成员;没有 const 修饰的成员函数,对数据成员则是可读可写
的。
除此之外,在类的成员函数后面加 const 还有什么好处呢?那就是常量(即 const)对象可以调用
const 成员函数,而不能调用非const修饰的函数。正如非const类型的数据可以给const类型的变量赋
值一样,反之则不成立。
#include
using namespace std;
class CStu
{
public:
int a;
CStu()
{
a = 12;
}
void Show() const
{
//a = 13; //常函数不能修改数据成员
cout <<a << “I am show()” << endl;
}
};
int main()
{
CStu st;
st.Show();
system(“pause”);
return 0;
}
1.3.20 说说什么是虚继承,解决什么问题,如何实现?
虚继承是 解决C++多重继承问题的一种手段, 从不同途径继承来的同一基类, 会在子类中存在多份拷
贝。 这将存在两个问题:其一,浪费存储空间;第二,存在二义性问题, 通常可以将派生类对象的地址
赋值给基类对象,实现的具体方式是,将基类指针指向继承类(继承类有基类的拷贝)中的基类对象的
地址,但是多重继承可能存在一个基类的多份拷贝,这就出现了二义性。
虚继承可以解决多种继承前面提到的两个问题
#include
using namespace std;
class A{
public:
int _a;
};
class B :virtual public A
{
public:int _b;
};
class C :virtual public A
{
public:
int _c;
};
class D :public B, public C
{
public:
int _d;
};
//菱形继承和菱形虚继承的对象模型
int main()
{
D d;
d.B::_a = 1;
d.C::_a = 2;
d._b = 3;
d._c = 4;
d._d = 5;
cout << sizeof(D) << endl;
return 0;
}
分别从菱形继承和虚继承来分析:
image-20220711120000285
菱形继承中A在B,C,D,中各有一份, 虚继承中,A共享。
上面的虚继承表 实际上是一个指针数组。B、C实际上是虚基表指针,指向虚基表。
虚基表:存放相对偏移量,用来找虚基类
1.3.21 简述一下虚函数和纯虚函数,以及实现原理
C++中的虚函数的作用主要是实现了多态的机制。
关于多态,简而言之就是用父类型的指针指向其子类的实例,然后通过父类的指针调用实际子类的成员函数。这种技术可以让父类的指针有“多种形态”,这是一种泛型技术。如果调用非虚函数,则无论实际对象是什么类型,都执行基类类型所
定义的函数。 非虚函数总是在编译时 根据调用该函数的对象,引用或指针的类型 而确定。如果调用
虚函数,则直到运行时才能确定调用哪个函数, 运行的虚函数是引用所绑定或指针所指向的对象所属类型定义的版本。虚函数必须是基类的非静态成员函数。虚函数的作用是实现动态联编,也就是在程序的运行阶段动态地选择合适的成员函数,在定义了虚函数后,可以在
基类的派生类中对虚函数重新定义,在派生类中重新定义的函数应与虚函数具有相同的形参个数和形参类型。以实现统一
的接口,不同定义过程。如果在派生类中 没有对虚函数重新定义,则它继承其基类的虚函数。
class Person{
public:
//虚函数
virtual void GetName(){
cout<<“PersonName:xiaosi”<<endl;
};
};
class Student:public Person{
public:
void GetName(){
cout<<“StudentName:xiaosi”<<endl;
};
};
int main(){
//指针
Person *person = new Student();
//基类调用子类的函数
person->GetName();//StudentName:xiaosi
}
虚函数(Virtual Function)是通过一张虚函数表(Virtual Table)来实现的。简称为V-Table。在
这个表中,主是要一个类的虚函数的地址表,这张表解决了继承、覆盖的问题,保证其容真实反应
实际的函数。这样,在有虚函数的类的实例中这个表被分配在了这个实例的内存中,所以,当我们
用父类的指针来操作一个子类的时候,这张虚函数表就显得由为重要了,它就像一个地图一样,指
明了实际所应该调用的函数。
纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现
方法。在基类中实现纯虚函数的方法是在函数原型后加“=0” virtualvoid GetName() =0。在很多情
况下,基类本身生成对象是不合情理的。例如,动物作为一个基类可以派生出老虎、孔雀等子类,
但动物本身生成对象明显不合常理。
为了解决上述问题,将函数定义为纯虚函数,则编译器要求在派生类中必须予以重写以实现多态性。同时含有纯虚拟函数的类称为抽象类,它不能生成对象。这样就很好地解决了上述两个问题。 将函数定义为纯虚函数能够说明,该函数为后代类型提供了可以
覆盖的接口, 但是这个类中的函数绝不会调用。 声明了纯虚函数的类是一个抽象类。所以,用户不能创建类的实例,只能创建它的派生类的实例。必须在继承类中重新声明函数(不要后面的=0)
否则该派生类也不能实例化, 而且它们在抽象类中往往没有定义。定义纯虚函数的目的在于,使派
生类仅仅只是继承函数的接口。纯虚函数的意义,让所有的类对象(主要是派生类对象)都可以执
行纯虚函数的动作,但类 无法为纯虚函数提供一个合理的缺省实现。 所以类纯虚函数的声明就是在
告诉子类的设计者,“你必须提供一个纯虚函数的实现,但我不知道你会怎样实现它”。
//抽象类
class Person{
public:
//纯虚函数
virtual void GetName()=0;
};
class Student:public Person{
public:
Student(){
};
void GetName(){ cout<<“StudentName:xiaosi”<<endl;
};
};
int main(){
Student student;
}
1.3.22 说说纯虚函数能实例化吗,为什么?派生类要实现吗,为什么?
参考回答
纯虚函数不可以实例化,但是可以用其派生类实例化,示例如下:
#include
using namespace std;
class Base
{
public:
virtual void func() = 0;
};
class Derived :public Base
{
public:
void func() override
{
cout << “哈哈” << endl;
}
};
int main()
{
Base *b = new Derived();
b->func();
虚函数的原理 采用 vtable。类中含有纯虚函数时,其vtable 不完全,有个空位。
即“纯虚函数在类的vftable表中对应的表项被赋值为0。也就是指向一个不存在的函数。由于编译
器绝对不允许有调用一个不存在的函数的可能,所以 该类不能生成对象。在它的派生类中,除非重
写此函数,否则也不能生成对象。”
所以纯虚函数不能实例化。
纯虚函数是在基类中声明的虚函数,它要求任何派生类都要定义自己的实现方法,以实现多态性。
定义纯虚函数是为了实现一个接口,用来规范派生类的行为,也即 规范继承这个类的程序员必须实
现这个函数。派生类仅仅只是继承函数的接口。纯虚函数的意义在于,让所有的类对象(主要是派
生类对象)都可以执行纯虚函数的动作,但基类无法为纯虚函数提供一个合理的缺省实现。所以类
纯虚函数的声明就是在告诉子类的设计者,“你必须提供一个纯虚函数的实现,但我不知道你会怎
样实现它”。
1.3.23 说说C++中虚函数与纯虚函数的区别
参考回答
虚函数和纯虚函数可以定义在同一个类中,含有纯虚函数的类被称为抽象类,而只含有虚函数的类
不能被称为抽象类。
虚函数 可以被直接使用,也可以被子类重载以后,以多态的形式调用 而纯虚函数必须在子类中实**
现该函数才可以使用,因为纯虚函数在基类有声明而没有定义。
虚函数和纯虚函数都可以在子类中被重载,以多态的形式被调用。
虚函数和纯虚函数 通常存在于抽象基类之中,被继承的子类重载,目的是提供一个统一的接口。
虚函数的定义形式: virtual{} ; 纯虚函数的定义形式: virtual
{ } = 0 ;在虚函数和纯虚函数的定义中不能有static标识符,原因很简单,被static修饰的函数在编译时要求前期绑定, 然而虚
函数却是动态绑定,而且被两者修饰的函数生命周期也不一样。
答案解析""
class A
{
public:
virtual void foo()
{
cout<<“A::foo() is called”<<endl;
}
};
class B:public A
{
public:
void foo()
{
cout<<“B::foo() is called”<<endl;
}
};
int main(void)
{
A *a = new B();
a->foo(); // 在这里,a虽然是指向A的指针,但是被调用的函数(foo)却是B的!
return 0;
}
这个例子是虚函数的一个典型应用,通过这个例子,也许你就对虚函数有了一些概念。它虚就虚在
所谓“推迟联编”或者“动态联编”上,一个类函数的调用 并不是在编译时刻被确定的, 而是在运行时
刻被确定的。由于编写代码的时候并不能确定被调用的是 基类的函数 还是哪个派生类的函数, 所以
被成为“虚”函数。
虚函数只能借助于指针或者引用来达到多态的效果。
纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现
方法。在基类中实现纯虚函数的方法是在函数原型后加“=0”
virtual void funtion1()=0
为了方便使用多态特性,我们常常需要在基类中定义虚拟函数。
在很多情况下,基类本身生成对象是不合情理的。例如,动物作为一个基类可以派生出老虎、孔雀
等子类,但动物本身生成对象明显不合常理。
为了解决上述问题,引入了纯虚函数的概念,将函数定义为纯虚函数(方法:virtual ReturnType
Function()= 0;),则 编译器要求 在派生类中必须予以重写以实现多态性。同时含有纯虚拟函数的
类称为抽象类,它不能生成对象。这样就很好地解决了上述两个问题。
声明了纯虚函数的类是一个抽象类。所以,用户不能创建类的实例,只能创建它的派生类的实例。
纯虚函数最显著的特征是:它们必须在继承类中重新声明函数(不要后面的=0,否则该派生类也
不能实例化),而且它们在抽象类中往往没有定义。
定义纯虚函数的目的在于,使派生类仅仅只是继承函数的接口。
纯虚函数的意义,让所有的类对象(主要是派生类对象)都可以执行纯虚函数的动作,但类无法为
纯虚函数提供一个合理的缺省实现。所以类纯虚函数的声明就是在告诉子类的设计者,“你必须提
供一个纯虚函数的实现,但我不知道你会怎样实现它”。
1.3.24 说说 C++ 中什么是菱形继承问题,如何解决
参考回答
下面的图表可以用来解释菱形继承问题。
image-20220711135230960
假设我们有类B和类C,它们都继承了相同的类A。另外我们还有类D,类D通过多重继承机制继承
了类B和类C。因为上述图表的形状类似于菱形,因此这个问题被形象地称为菱形继承问题。现
在,我们将上面的图表翻译成具体的代码:
/*
Animal类对应于图表的类A
/
class Animal { / … /
}; // 基类
{
int weight;
public:
int getWeight() { return weight; }
};
class Tiger : public Animal { / … / };
class Lion : public Animal { / … /
}
class Liger : public Tiger, public Lion { / … */ }
在上面的代码中,我们给出了一个具体的菱形继承问题例子。Animal类对应于最顶层类(图表中
的A),Tiger和Lion分别对应于图表的B和C,Liger类(狮虎兽,即老虎和狮子的杂交种)对应于
D。
现在,问题是如果我们有这种继承结构会出现什么样的问题。=
int main( )
{
Liger lg;
/*编译错误,下面的代码不会被任何C++编译器通过 */
int weight = lg.getWeight();
}
在我们的继承结构中,我们可以看出Tiger和Lion类都继承自Animal基类。所以问题是:因为Liger
多重继承了Tiger和Lion类,因此Liger类会有两份Animal类的成员(数据和方法),Liger对
象"lg"会包含Animal基类的两个子对象。
所以,你会问Liger对象有两个Animal基类的子对象会出现什么问题?再看看上面的代码-调
用"lg.getWeight()"将会导致一个编译错误。这是因为编译器并不知道是调用Tiger类的getWeight()
还是调用Lion类的getWeight()。所以,调用getWeight方法是不明确的,因此不能通过编译。
我们给出了菱形继承问题的解释,但是现在我们要给出一个菱形继承问题的解决方案。如果Lion类
和Tiger类在分别继承Animal类时都用virtual来标注,对于每一个Liger对象,C++会保证只有一个
Animal类的子对象会被创建。看看下面的代码:
class Tiger : virtual public Animal { /* … / };
class Lion : virtual public Animal { / … */ };
你可以看出唯一的变化就是我们在类Tiger和类Lion的声明中增加了"virtual"关键字。现在类Liger
对象将会只有一个Animal子对象,下面的代码编译正常:
int main( )
{
Liger lg;
/*既然我们已经在Tiger和Lion类的定义中声明了"virtual"关键字,于是下面的代码编译OK
*/
int weight = lg.getWeight();
}
1.3.25 请问构造函数中的能不能调用虚方法
参考回答
不要在构造函数中调用虚方法,从语法上讲,调用完全没有问题,但是从效果上看,往往不能达到
需要的目的。
派生类对象 构造期间 进入基类的构造函数时,对象类型变成了基类类型,而不是派生类类型。
同样,进入基类析构函数时,对象也是基类类型。
所以,虚函数 始终仅仅调用 基类的虚函数(如果是基类调用虚函数), 不能达到多态的效果 ,所以
放在构造函数中是没有意义的,而且往往不能达到本来想要的效果。
1.3.26 请问拷贝构造函数的参数是什么传递方式,为什么
参考回答
拷贝构造函数的参数必须使用引用传递
如果拷贝构造函数中的参数不是一个引用,即形如CClass(const CClass c_class),那么就相当于采
用了传值的方式(pass-by-value),而 传值的方式会 调用该类的拷贝构造函数, 从而造成无穷递归地
调用拷贝构造函数。因此拷贝构造函数的参数必须是一个引用。
需要澄清的是, 传指针 其实也是 传值,如果上面的拷贝构造函数写成CClass(const CClass*
c_class), 也是不行的。事实上,只有传引用不是传值外, 其他所有的传递方式都是传值。
1.3.27 说说类方法和数据的权限有哪几种
参考回答
C++通过 public、protected、private 三个关键字来控制成员变量和成员函数的访问权限,它们分
别表示公有的、受保护的、私有的,被称为成员访问限定符
image-20220711140153088
下面介绍一个例子。
class Person
{
public:
Person(const string& name, int age) : m_name(name), m_age(age)
{
}
void ShowInfo()
{
cout << “姓名:” << m_name << endl; cout << “年龄:” << m_age << endl;
}
protected:
string m_name; //姓名
private:
int m_age; //年龄
};
子类:
class Teacher : public Person
{
public:
Teacher(const string& name, int age, const string& title)
: Person(name, age), m_title(title)
{
}
void ShowTeacherInfo()
{
ShowInfo(); //正确,public属性子类可见
cout << “姓名:” << m_name << endl; //正确,protected属性子类可见
cout << “年龄:” << m_age << endl; //错误,private属性子类不可见
cout << “职称:” << m_title << endl; //正确,本类中可见自己的所有成员
}
private:
string m_title; //职称
};
调用方:
void test()
{
Person person(“张三”, 22);
person.ShowInfo(); //public属性,对外部可见
cout << person.m_name << endl; //protected属性,对外部不可见
cout << person.m_age << endl; //private属性,对外部不可见
}
1.3.28 如何理解抽象类?
参考回答
抽象类的定义如下:
纯虚函数 是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现
方法。在基类中实现纯虚函数的方法是在函数原型后加“=0”,有虚函数的类就叫做抽象类。
抽象类有如下几个特点:
1)抽象类 只能用作其他类的基类,不能建立抽象类对象。
2)抽象类 不能用作 参数类型、函数返回类型或显式转换的类型。
3)可以定义指向抽象类的指针和引用,此指针可以指向它的派生类,进而实现多态性。
1.3.29 什么是多态?除了虚函数,还有什么方式能实现多态?
参考回答
多态是面向对象的重要特性之一,它是一种行为的封装,就是不同对象对同一行为会有不同的状
态。(举例 : 学生和成人都去买票时,学生会打折,成人不会)
多态是以封装和继承为基础的。在C++中多态分为 静态多态(早绑定)和动态多态(晚绑定)两
种,其中动态多态是通过虚函数实现,静态多态 通过函数重载实现,代码如下:
class A
{
public:
void do(int a);
void do(int a, int b);
};
1.3.30 简述一下虚析构函数,什么作用
虚析构函数,是将基类的析构函数声明为virtual,举例如下:
class TimeKeeper
{
public:
TimeKeeper() {}
virtual ~TimeKeeper() {}
};
虚析构函数的主要作用 是防止内存泄露。
定义一个基类的指针p,在delete p时,如果基类的析构函数是虚函数,这时只会看p所赋值的对
象,如果p赋值的对象是派生类的对象,就会调用派生类的析构函数(毫无疑问,在这之前也会先
调用基类的构造函数,在调用派生类的构造函数,然后调用派生类的析构函数,基类的析构函数,
所谓先构造的后释放); 如果p赋值的对象是基类的对象,就会调用基类的析构函数,这样就不会
造成内存泄露。
如果基类的析构函数不是虚函数,在delete p时,调用析构函数时,只会看指针的数据类型,而
不会去看赋值的对象,这样就会造成内存泄露。
答案解析
我们创建一个TimeKeeper基类和一些及其它的派生类作为不同的计时方法
class TimeKeeper
{
public:
TimeKeeper() {}
~TimeKeeper() {} //非virtual的
};
//都继承与TimeKeeper
class AtomicClock :public TimeKeeper{};
class WaterClock :public TimeKeeper {};
class WristWatch :public TimeKeeper {};
如果客户想要在程序中使用时间,不想操作时间如何计算等细节,这时候我们可以设计
factory(工厂)函数,让函数返回指针指向一个计时对象。该函数返回一个基类指针,这个基类
指针是指向于派生类对象的
TimeKeeper* getTimeKeeper()
{
//返回一个指针,指向一个TimeKeeper派生类的动态分配对象
}
因为函数返回的对象存在于堆中, 因此为了在不使用时 我们需要使用释放该对象(delete)
TimeKeeper* ptk = getTimeKeeper();
delete ptk;
此处基类的析构函数是非virtual的,因此通过一个基类指针删除派生类对象是错误的
解决办法: 将 基类的析构函数改为virtual就正确了
class TimeKeeper
{
public:
TimeKeeper() {}
virtual ~TimeKeeper() {}
};
声明为virtual之后,通过基类指针删除派生类对象就会释放整个对象(基类+派生类)
1.3.31 说说什么是虚基类,可否被实例化?
参考回答
在 被继承的类前面加上virtual关键字,这时被继承的类称为虚基类,代码如下:
class A
class B1:public virtual A;
class B2:public virtual A;
class D:public B1,public B2;
虚继承的类可以被实例化,举例如下:
class Animal {/* … / };
class Tiger : virtual public Animal { / … / };
class Lion : virtual public Animal { / … */ }
int main( )
{
Liger lg;
/*既然我们已经在Tiger和Lion类的定义中声明了"virtual"关键字,于是下面的代码编译OK
*/
int weight = lg.getWeight();
}
1.3.32 简述一下拷贝赋值和移动赋值?
参考回答
拷贝赋值 是通过 拷贝构造函数来赋值,在创建对象时,使用同一类中之前创建的对象来初始化新创
建的对象。
移动赋值是通过移动构造函数来赋值,二者的主要区别在于
1)拷贝构造函数的形参 是一个 左值引用,而移动构造函数的形参是一个右值引用;
2)拷贝构造函数 完成的是 整个对象或变量的拷贝, 而移动构造函数是生成一个指针 指向源对象或
变量的地址,接管源对象的内存 ,相对于大量数据的拷贝节省时间和内存空间。
1.3.33仿函数了解吗?有什么作用
参考回答
仿函数(functor)又称为函数对象(function object)是一个能行使函数功能的类。仿函数的语
法几乎和我们普通的函数调用一样,不过作为仿函数的类,都必须重载operator()运算符,举个例
子:
class Func{
public:
void operator() (const string& str) const {
cout<<str<<endl;
}
};
Func myFunc;
myFunc(“helloworld!”);
helloworld!
仿函数既能想普通函数一样传入给定数量的参数,还能存储或者处理更多我们需要的有用信息。我
们可以举个例子:
假设有一个 vector ,你的任务是统计长度小于5的string的个数,如果使用 count_if
函数的话,你的代码可能长成这样:
bool LengthIsLessThanFive(const string& str) {
return str.length()<5;
}
int res=count_if(vec.begin(), vec.end(), LengthIsLessThanFive);
其中 count_if 函数的第三个参数是一个函数指针,返回一个bool类型的值。一般的,如果需要将
特定的阈值长度也传入的话,我们可能将函数写成这样:
bool LenthIsLessThan(const string& str, int len) {
return str.length()<len;
}
这个函数看起来比前面一个版本更具有一般性,但是他不能满足 count_if 函数的参数要求:
count_if 要求的是unary function(仅带有一个参数)作为它的最后一个参数。如果我们使用仿
函数,是不是就豁然开朗了呢:
class ShorterThan {
public:
explicit ShorterThan(int maxLength) : length(maxLength) {}
bool operator() (const string& str) const {
return str.length() < length;
}
private:
const int length;
};
1.3.34 C++ 中哪些函数不能被声明为虚函数?
参考回答
常见的不能声明为虚函数的有:普通函数(非成员函数),静态成员函数,内联成员函数,构造函
数,友元函数。
为什么C++不支持普通函数为虚函数?
普通函数(非成员函数)只能被overload,不能被override, 声明为虚函数也没有什么意思,因此
编译器会在编译时绑定函数。
为什么C++不支持构造函数为虚函数?
这个原因很简单,主要是从语义上考虑,所以不支持。因为构造函数本来就是为了明确初始化对象
成员才产生的,然而virtual function主要是为了 再不完全了解细节的情况下也能正确处理对象。
另外,virtual函数是在不同类型的对象产生不同的动作,现在对象还没有产生, 如何使用virtual函
数来完成你想完成的动作。(这不就是典型的悖论)
构造函数用来创建一个新的对象,而虚函数的运行是建立在对象的基础上,在构造函数执行时,对象尚
未形成,所以不能将构造函数定义为虚函数
为什么C++不支持内联成员函数为虚函数?
其实很简单,那内联函数就是为了在代码中直接展开,减少函数调用花费的代价,虚函数是为了在
继承后对象能够准确的执行自己的动作,这是不可能统一的。(再说了,inline 函数在编译时被展
开,虚函数在运行时才能动态的绑定函数 )
内联函数是在编译时期展开,而虚函数的特性是运行时才动态联编,所以两者矛盾,不能定义内联函数
为虚函数
为什么C++不支持静态成员函数为虚函数?
这也很简单,静态成员函数对于每个类来说只有一份代码,所有的对象都共享这一份代码,他也没
有要动态绑定的必要性。
静态成员函数属于一个类而非某一对象,没有this指针,它无法进行对象的判别
为什么C++不支持友元函数为虚函数?****
因为C++不支持友元函数的继承,对于 没有继承特性的函数没有虚函数的说法。
1.3.35 解释下 C++ 中类模板和模板类的区别
参考回答
类模板 是 模板的定义,不是一个实实在在的类, 定义中用到通用类型参数
模板类 是 实实在在的类定义,是类模板的实例化。类定义中参数被实际类型所代替
答案解析
类模板的类型参数可以有一个或多个,每个类型前面都必须加class,如template <class T1,class
T2>class someclass{…}; 在定义对象时分别代入实际的类型名,如 someclass<int,double> obj;
和 使用类一样,使用类模板时要注意其作用域,只能在其有效作用域内用它定义对象。
模板可以有层次,一个类模板可以作为基类,派生出 派生模板类
1.3.36 虚函数表里存放的内容是什么时候写进去的?
参考回答
虚函数表 是一个 存储虚函数地址的数组,以NULL结尾。虚表(vftable)在编译阶段生成, 对象内存
空间开辟以后,写入对象中的 vfptr,然后调用构造函数。即:虚表 在构造函数之前写入
除了在构造函数之前写入之外,我们还需要考虑到虚表的二次写入机制,通过此机制让每个对象的
虚表指针都能准确的指向到自己类的虚表,为实现动多态提供支持。
1.4 STL
1.4.1 请说说 STL 的基本组成部分
参考回答
标准模板库(Standard Template Library,简称STL)简单说,就是一些常用数据结构和算法的模板的
集合。
广义上讲,STL分为3类:Algorithm(算法)、Container(容器)和Iterator(迭代器),容器和算
法通过迭代器可以进行无缝地连接。
详细的说,STL由6部分组成:容器(Container)、算法(Algorithm)、 迭代器(Iterator)、仿函数
(Function object)、适配器(Adaptor)、空间配制器(Allocator)。
答案解析
标准模板库STL主要由6大组成部分:
容器(Container)
是一种数据结构, 如list, vector, 和deques,以模板类的方法提供。为了访问容器中的数据,可
以使用由容器类输出的迭代器。
算法(Algorithm)
是用来操作容器中的数据的模板函数。例如,STL用sort()来对一 个vector中的数据进行排序,
用find()来搜索一个list中的对象, 函数本身与他们操作的数据的结构和类型无关,因此他们可以
用于从简单数组到高度复杂容器的任何数据结构上。
迭代器(Iterator)
提供了访问容器中对象的方法。例如,可以使用一对迭代器指定list或vector中的一定范围的对
象。 迭代器就如同一个指针。事实上,C++ 的指针也是一种迭代器。 但是,迭代器也可以是那些
定义了operator*()以及其他类似于指针的操作符方法的类对象;
仿函数(Function object)
仿函数又称之为函数对象, 其实就是重载了操作符的struct,没有什么特别的地方。
适配器(Adaptor)
简单的说就是一种接口类,专门用来修改现有类的接口,提供一中新的接口;或调用现有的函
数来实现所需要的功能。主要包括3中适配器Container Adaptor、Iterator Adaptor、Function
Adaptor。
空间配制器(Allocator)
为STL提供空间配置的系统。其中主要工作包括两部分:
(1)对象的创建与销毁;
(2)内存的获取与释放。
1.4.2 请说说 STL 中常见的容器,并介绍一下实现原理
参考回答
容器可以用于存放各种类型的数据(基本类型的变量,对象等)的数据结构,都是模板类,分为顺序
容器、关联式容器、容器适配器三种类型,三种类型容器特性分别如下:
顺序容器容器并非排序的,元素的插入位置同元素的值无关。包含vector、deque、list,具体实现原理
如下:
(1)vector 头文件
动态数组。元素在内存连续存放。随机存取任何元素都能在常数时间完成。在尾端增删元
素具有较佳的性能。
(2)deque 头文件
双向队列。元素在内存连续存放。随机存取任何元素都能在常数时间完成(仅次于vector)。在
两端增删元素具有较佳的性能(大部分情况下是常数时间)。
(3)list 头文件
双向链表。元素在内存不连续存放。在任何位置增删元素都能在常数时间完成。不支持随机
存取。
关联式容器
元素是排序的;插入任何元素,都按相应的排序规则来确定其位置;在查找时具有非常好的性
能;通常以平衡二叉树的方式实现。包含set、multiset、map、multimap,具体实现原理如下:
(1)set/multiset 头文件
set 即集合。set中不允许相同元素,multiset中允许存在相同元素。
(2)map/multimap 头文件
map与set的不同在于map中存放的元素有且仅有两个成员变,一个名为first,另一个名为
second, map根据first值对元素从小到大排序,并可快速地根据first来检索元素。
注意:map同multimap的不同在于是否允许相同first值的元素。
容器适配器
封装了一些基本的容器,使之具备了新的函数功能,比如把deque封装一下变为一个具有stack
功能的数据结构。这新得到的数据结构就叫适配器。包含stack,queue,priority_queue,具体实现
原理如下:
(1)stack 头文件
栈是项的有限序列,并满足序列中被删除、检索和修改的项只能是最进插入序列的项(栈顶的项)。后进先出。
(2)queue 头文件
队列。插入只可以在尾部进行,删除、检索和修改只允许从头部进行。先进先出。
(3)priority_queue
头文件
优先级队列。内部维持某种有序,然后确保优先级最高的元素总是位于头部。最高优先级元
素总是第一个出列。
1.4.3 说说 STL 中 map hashtable deque list 的实现原理
map、hashtable、deque、list实现机理分别为红黑树、函数映射、双向队列、双向链表,他们的特
性分别如下:
map实现原理
map内部实现了一个红黑树(红黑树是非严格平衡的二叉搜索树,而AVL是严格平衡二叉搜索
树), 红黑树有自动排序的功能,因此map内部所有元素都是有序的,红黑树的每一个节点都代
表着map的一个元素。因此,对于map进行的查找、删除、添加等一系列的操作都相当于是对红
黑树进行的操作。map中的元素 是按照二叉树(又名二叉查找树、二叉排序树)存储的, 特点就是左子树上所有节点的键值都小于根节点的键值,右子树所有节点的键值都大于根节点的键值。使用中序遍历可将键值按照从小到大遍历出来。
hashtable(也称散列表,直译作哈希表)实现原理
hashtable采用了函数映射的思想 记录的存储位置 与 记录的关键字关联起来,从而能够很快速地
进行查找。这决定了哈希表特殊的数据结构,它同数组、链表以及二叉排序树等相比较有很明显的
区别,它能够快速定位到想要查找的记录,而不是 与表中存在的记录的关键字进行比较来进行查
找。
deque实现原理
deque内部实现的是一个双向队列。元素在内存连续存放。随机存取任何元素都在常数时间完成
(仅次于vector)。所有适用于vector的操作都适用于deque。在两端增删元素具有较佳的性能
(大部分情况下是常数时间)。
list实现原理
list内部实现的是一个双向链表。元素在内存不连续存放。在任何位置增删元素都能在常数时间
完成。不支持随机存取。无成员函数,给定一个下标i,访问第i个元素的内容,只能从头部挨个遍
历到第i个元素。
1.4.4 请你来介绍一下 STL 的空间配置器(allocator)
参考回答
一般情况下,一个程序包括数据结构和相应的算法,而数据结构作为存储数据的组织形式,与内存空间
有着密切的联系。在C++ STL中,空间配置器便是用来实现内存空间(一般是内存,也可以是硬盘等空间)
分配的工具,他与容器联系紧密,每一种容器的空间分配都是通过空间分配器alloctor实现的。
答案解析
两种C++类对象实例化方式的异同
在c++中,创建类对象一般分为两种方式:一种是直接利用构造函数,直接构造类对象,如 Test
test(); 另一种是通过new来实例化一个类对象,如 Test *pTest = new Test;那么,这两种方式有
什么异同点呢?
我们知道,内存分配主要有三种方式:
(1) 静态存储区分配: 内存在程序编译的时候已经分配好,这块内存在程序的整个运行空间内都
存在。如全局变量,静态变量等。
(2) 栈空间分配: 程序在运行期间,函数内的局部变量通过栈空间来分配存储(函数调用栈),
当函数执行完毕返回时,相对应的栈空间被立即回收。主要是局部变量。
(3)堆空间分配 :程序在运行期间,通过在堆空间上为数据分配存储空间,通过malloc和new创
建的对象都是从堆空间分配内存,这类空间需要程序员自己来管理,必须通过free()或者是delete()
函数对堆空间进行释放,否则会造成内存溢出。
那么,从内存空间分配的角度来对这两种方式的区别,就比较容易区分:
(1)对于第一种方式来说,是直接通过调用Test类的构造函数来实例化Test类对象的,如果该实例
化对象是一个局部变量,则其是在栈空间分配相应的存储空间。
(2)对于第二种方式来说,就显得比较复杂。这里主要以new类对象来说明一下。new一个类对象,
其实是执行了两步操作:首先,调用new在堆空间分配内存,然后调用类的构造函数构造对象的内
容;同样, 使用delete释放时,也是经历了两个步骤: 首先调用类的析构函数释放类对象,然后 调
用delete释放堆空间。
C++ STL空间配置器实现
很容易想象,为了实现空间配置器,完全可以利用new和delete函数 并对其进行封装实现STL的
空间配置器,的确可以这样。但是,为了最大化提升效率,SGI STL版本 并没有简单的这样做,而
是采取了一定的措施,实现了更加高效复杂的空间分配策略。由于以上的构造都分为两部分,所
以,在SGI STL中,将对象的构造切分开来,分成空间配置和对象构造两部分。
内存配置操作: 通过alloc::allocate()实现
内存释放操作: 通过alloc::deallocate()实现
对象构造操作: 通过::construct()实现
对象释放操作: 通过::destroy()实现
关于内存空间的配置与释放,SGI STL采用了两级配置器:
一级配置器主要是考虑大块内存空间,利用 malloc和free实现;
二级配置器主要是考虑小块内存空间而设计的(为了最大化解决内存
碎片问题,进而提升效率),采用链表free_list来维护内存池(memory pool),free_list通过**
union结构实现,空闲的内存块 互相挂接在一块,内存块一旦被使用,则被从链表中剔除,易于维
护。
1.4.5 STL 容器用过哪些,查找的时间复杂度是多少,为什么?
参考回答
STL中常用的容器有vector、deque、list、map、set、multimap、multiset、unordered_map、
unordered_set等。容器底层实现方式及时间复杂度分别如下:
vector
采用一维数组实现,元素在内存连续存放,不同操作的时间复杂度为:
插入: O(N)
查看: O(1)
删除: O(N)
deque
采用双向队列实现,元素在内存连续存放,不同操作的时间复杂度为:
插入: O(N)
查看: O(1)
删除: O(N)
list
采用双向链表实现,元素存放在堆中,不同操作的时间复杂度为:
插入: O(1)
查看: O(N)
删除: O(1)
map、set、multimap、multiset
上述四种容器采用红黑树实现,红黑树是平衡二叉树的一种。不同操作的时间复杂度近似为:
插入: O(logN)
查看: O(logN)
删除: O(logN)
unordered_map、unordered_set、unordered_multimap、 unordered_multiset
上述四种容器采用哈希表实现,不同操作的时间复杂度为:
插入: O(1),最坏情况O(N)
查看: O(1),最坏情况O(N)
删除: O(1),最坏情况O(N)
注意:容器的时间复杂度取决于其底层实现方式。
1.4.6 迭代器用过吗?什么时候会失效?
参考回答
用过,常用容器迭代器失效情形如下。
对于序列容器vector,deque来说,使用erase后,后边的每个元素的迭代器都会失效,后边每个
元素都往前移动一位,erase返回下一个有效的迭代器。
对于关联容器map,set来说,使用了erase后,当前元素的迭代器失效,但是其结构是红黑树,
删除当前元素,不会影响下一个元素的迭代器,所以在调用erase之前,记录下一个元素的迭代器
即可。
对于list来说,它使用了不连续分配的内存,并且它的erase方法也会返回下一个有效的迭代器,因
此上面两种方法都可以使用。****
1.4.7 说一下STL中迭代器的作用,有指针为何还要迭代器?
参考回答
迭代器的作用
(1)用于指向 顺序容器和关联容器中的元素
(2)通过迭代器可以读取它指向的元素
(3)通过非const迭代器还可以修改 其指向的元素
迭代器和指针的区别
迭代器不是指针,是类模板,表现的像指针。他只是模拟了指针的一些功能,重载了指针的一
些操作符,–>、++、–等。迭代器封装了指针,是一个”可遍历STL( Standard Template
Library)容器内全部或部分元素”的对象,本质是封装了原生指针,是指针概念的一种提升,提供
了比指针更高级的行为,相当于一种智能指针,他可以根据不同类型的数据结构来实现不同的
++,–等操作。
迭代器返回的是对象引用而不是对象的值,所以cout只能输出迭代器使用取值后的值 而不能直
接输出其自身。
迭代器产生的原因
Iterator类的访问方式就是把不同集合类的访问逻辑抽象出来,使得不用暴露集合内部的结构而
达到循环遍历集合的效果。
答案解析
1.迭代器
Iterator(迭代器)模式 又称 游标(Cursor)模式,用于提供一种方法 顺序访问一个聚合对象中
各个元素, 而又不需暴露该对象的内部表示。 或者这样说可能更容易理解:Iterator模式是运用于
聚合对象的一种模式,通过运用该模式,使得我们可以在不知道对象内部表示的情况下,按照一定
顺序(由iterator提供的方法)访问聚合对象中的各个元素。 由于Iterator模式的以上特性:与聚
合对象耦合,在一定程度上限制了它的广泛运用,一般仅用于底层聚合支持类,如STL的list、
vector、stack等容器类及ostream_iterator等扩展Iterator。
- 迭代器示例:#include
#include
using namespace std;
int main() {
vector v; //一个存放int元素的数组,一开始里面没有元素
v.push_back(1);
v.push_back(2);
v.push_back(3);
v.push_back(4);
vector::const_iterator i; //常量迭代器
for (i = v.begin(); i != v.end(); ++i) //v.begin()表示v第一个元素迭代器指针,++i
指向下一个元素
cout << *i << “,”; //*i表示迭代器指向的元素
cout << endl;
vector::reverse_iterator r; //反向迭代器
for (r = v.rbegin(); r != v.rend(); r++)
cout << *r << “,”;
cout << endl;
vector::iterator j; //非常量迭代器
for (j = v.begin();j != v.end();j++)
*j = 100;
for (i = v.begin();i != v.end();i++)
cout << i << “,”;
return 0;
}
/
运行结果:
1,2,3,4,
4,3,2,1,
100,100,100,100,
*
1.4.8 说说 STL 迭代器是怎么删除元素的
参考回答]
这是主要考察迭代器失效的问题。
对于 序列容器vector,deque来说,使用erase后, 后边的每个元素的迭代器都会失效,后边每个
元素都往前移动一位,erase返回下一个有效的迭代器;
对于关联容器map,set来说,使用了erase后,当前元素的迭代器失效,但是其结构是红黑树,
删除当前元素,不会影响下一个元素的迭代器,所以在调用erase之前,记录下一个元素的迭代器
即可;
对于list来说,它使用了不连续分配的内存,并且它的erase方法也会返回下一个有效的迭代器,因
此上面两种方法都可以使用。
答案解析
容器上迭代器分类如下表(详细实现过程请翻阅相关资料详细了解):
image-20220711161109510
1.4.9 说说 STL 中 resize 和 reserve 的区别
参考回答
首先必须弄清楚两个概念:
(1)capacity:该值在容器初始化时赋值,指的是容器能够容纳的最大的元素的个数。还不能通
过下标等访问,因为此时容器中还没有创建任何对象。
(2)size:指的是此时容器中实际的元素个数。可以通过下标访问0-(size-1)范围内的对象。
resize和reserve区别主要有以下几点:
(1)resize既分配了空间,也创建了对象;reserve表示容器预留空间,但并不是真正的创建对
象,需要通过insert()或push_back()等创建对象。
(2)resize既修改capacity大小,也修改size大小;reserve只修改capacity大小,不修改size大
小。
(3)两者的形参个数不一样。 resize带两个参数,一个表示容器大小,一个表示初始值(默认为
0);reserve只带一个参数,表示容器预留的大小。
答案解析
问题延伸:
resize 和 reserve 既有差别,也有共同点。两个接口的共同点是它们都保证了vector的空间大小
(capacity)最少达到它的参数所指定的大小。下面就他们的细节进行分析。
为实现resize的语义,resize接口做了两个保证:
(1)保证区间[0, new_size)范围内数据有效,如果下标index在此区间内,vector[indext]是合法
的;
(2)保证区间[0, new_size)范围以外数据无效,如果下标index在区间外,vector[indext]是非法
的。
reserve只是保证vector的空间大小(capacity)最少达到它的参数所指定的大小n。在区间[0, n)范围
内,如果下标是index,vector[index]这种访问有可能是合法的,也有可能是非法的,视具体情况而定。
以下是两个接口的源代码:
void resize(size_type new_size)
{
resize(new_size, T());
}
void resize(size_type new_size, const T& x)
{
if (new_size < size())
erase(begin() + new_size, end()); // erase区间范围以外的数据,确保区间以外的数
据无效
else
insert(end(), new_size - size(), x); // 填补区间范围内空缺的数据,确保区间内的
数据有效
}
#include
#include
using namespace std;
int main()
{
vector a;
cout<<“initial capacity:”<<a.capacity()<<endl;
cout<<“initial size:”<<a.size()<<endl;
/resize改变capacity和size/
a.resize(20);
cout<<“resize capacity:”<<a.capacity()<<endl;
cout<<“resize size:”<<a.size()<<endl;
vector b;
/reserve改变capacity,不改变resize/
b.reserve(100);
cout<<“reserve capacity:”<<b.capacity()<<endl;
cout<<“reserve size:”<<b.size()<<endl;
return 0;
}
/*
运行结果:
initial capacity:0
initial size:0
resize capacity:20
resize size:20
reserve capacity:100
reserve size:0
*/
注意:如果n大于当前的vector的容量(是容量,并非vector的size),将会引起自动内存分配。所以现
有的pointer,references,iterators将会失效。而内存的重新配置会很耗时间。
1.4.10 说说 STL 容器动态链接可能产生的问题?
参考回答
可能产生 的问题
容器是一种 动态分配内存空间的一个变量集合类型变量。在一般的程序函数里,局部容器,参
数 传递容器,参数传递 容器的引用,参数传递 容器指针都是可以正常运行的,而在 动态链接库函数
内部使用容器也是没有问题的, 但是给 动态库函数 传递容器的对象本身,则会出现内存堆栈破坏的
问题。
产生问题的原因
容器和动态链接库相互支持不够好,动态链接库函数 中使用容器时,参数中只能传递容器的引用,
并且要保证容器的大小 不能超出初始大小,否则导致容器自动重新分配,就会出现内存堆栈破坏问
题。
1.4.11 说说 map 和 unordered_map 的区别?底层实现
参考回答
map和unordered_map的区别在于他们的实现基理不同。
map实现机理
map内部实现了一个红黑树(红黑树是非严格平衡的二叉搜索树,而AVL是严格平衡二叉搜索
树), 红黑树有自动排序的功能,因此map内部所有元素都是有序的,红黑树的每一个节点都代
表着map的一个元素。因此,对于map进行的查找、删除、添加等一系列的操作都相当于是对红
黑树进行的操作。 map中的元素是按照二叉树(又名二叉查找树、二叉排序树)存储的,特点就
是左子树上所有节点的键值都小于根节点的键值,右子树所有节点的键值都大于根节点的键值。使
用中序遍历可将键值按照从小到大遍历出来。
unordered_map实现机理
unordered_map内部实现了一个哈希表(也叫散列表),通过 把关键码值映 射到Hash表中一个位
置来访问记录,查找时间复杂度可达O(1),其中在海量数据处理中有着广泛应用。因此,元素的排列顺序是无序的。
1.4.12 说说 vector 和 list 的区别,分别适用于什么场景?
参考回答
vector和list区别在于底层实现机理不同,因而特性和适用场景也有所不同。
vector:一维数组
特点: 元素在内存连续存放,动态数组,在堆中分配内存,元素连续存放,有保留内存,如果减少大
小后内存也不会释放。
优点: 和数组类似开辟一段连续的空间,并且支持随机访问,所以它的 查找效率高其时间复杂度
O(1)。
缺点 :由于开辟一段连续的空间,所以插入删除会需要对数据进行移动比较麻烦,时间复杂度
O(n),另外当空间不足时还需要进行扩容。
list:双向链表
特点: 元素在堆中存放,每个元素都是存放在一块内存中,它的内存空间可以是不连续的,通过指针
来进行数据的访问。
优点: 底层实现是循环双链表,当对大量数据进行插入删除时,其时间复杂度O(1)。
缺点: 底层没有连续的空间,只能通过指针来访问,所以查找数据需要遍历其时间复杂度O(n),
没有提供[]操作符的重载
应用场景
vector拥有一段连续的内存空间,因此支持随机访问,如果需要高效的随即访问,而不在乎插入和删
除的效率,使用vector。
list拥有一段不连续的内存空间,如果需要高效的插入和删除,而不关心随机访问,则应使用list。
1.4.13 简述 vector 的实现原理
参考回答
vector底层实现原理为一维数组(元素在空间连续存放)。
新增元素
Vector 通过一个连续的数组存放元素,如果集合已满,在新增数据的时候,就要分配一块更大
的内存,将原来的数据复制过来,释放之前的内存,在插入新增的元素。插入新的数据分在 最后插
入push_back和 通过迭代器在任何位置插入,这里说一下通过迭代器插入,通过迭代器与第一个元
素的距离知道要插入的位置,即int index=iter-begin()。这个元素后面的所有元素都向后移动一个
位置,在空出来的位置上存入新增的元素。
//新增元素
void insert(const_iterator iter,const T& t )
{
int index=iter-begin();
if (index<size_)
{
if (size_==capacity_)
{
int capa=calculateCapacity();
newCapacity(capa);
}
memmove(buf+index+1,buf+index,(size_-index)*sizeof(T));
buf[index]=t;
size_++;
}
}
删除元素
删除和新增差不多,也分两种,删除最后一个元素pop_back和 通过迭代器删除任意一个元素
erase(iter)。 通过迭代器删除还是先找到要删除元素的位置,即int index=iter-begin(); 这个位置后
面的每个元素都想前移动一个元素的位置。同时我们知道erase不释放内存只初始化成默认值。
删除全部元素clear:只是循环调用了erase,所以删除全部元素的时候,不释放内存。内存是在
析构函数中释放的。
//删除元素
iterator erase(const_iterator iter)
{
int index=iter-begin();
if (index<size_ && size_>0)
{
memmove(buf+index ,buf+index+1,(size_-index)*sizeof(T));
buf[–size_]=T();
}
return iterator(iter);
}
迭代器iteraotr
迭代器iteraotr是STL的一个重要组成部分,通过iterator可以很方便的存储集合中的元素.STL为每个
集合都写了一个迭代器, 迭代器其实是对一个指针的包装,实现一些常用的方法,如++,–,!=,==,*,->等,
通过这些方法可以找到当前元素或是别的元素. vector是STL集合中比较特殊的一个,因为vector中
的每个元素都是连续的,所以在自己实现vector的时候可以用指针代替。
//迭代器的实现
template<class _Category,class _Ty,class _Diff = ptrdiff_t,class _Pointer =
_Ty *,class _Reference = _Ty&>
struct iterator
{
// base type for all iterator classes
typedef _Category iterator_category;
typedef _Ty value_type;
typedef _Diff difference_type;
typedef _Diff distance_type; // retained
typedef _Pointer pointer;
typedef _Reference reference;
};
vector实现源码
#ifndef CVECTOR_H
#define CVECTOR_H
namespace cth
{
class NoCopy
{
public:
inline NoCopy(){}
NoCopy(const NoCopy&);
NoCopy& operator=(const NoCopy&);
};
template
class viterator:public std::iteratorstd::forward_iterator_tag,T
{
public:
viterator()
{
t=NULL;
}
viterator(T* t_)
{
t=t_;
}
viterator(const viterator& other)
{
t=other.t;
}
viterator& operator=(const viterator& other)
{
t=other.t;
return *this;
}
viterator& operator++()
{
t++;
return *this;
}
viterator operator++(int)
{
viterator iter=*this;
t++;
return iter;
}
viterator operator+(int count)
{
viterator iter=*this;
iter.t+=count;
return iter;
}
viterator& operator–()
{
t–;
return *this;
}
viterator operator–(int)
{
viterator iter=*this;
t–;
return iter;
}
viterator operator-(int count)
{
viterator iter=*this;
iter.t-=count;
return iter;
}
int operator-(const viterator& other)
{
return t-other.t;
}
int operator-(const viterator& other)const
{
return t-other.t;
} T& operator*()
{
return *t;
}
const T& operator*() const
{
return *t;
}
T* operator->()
{
return t;
}
const T* operator->() const
{
return t;
}
inline bool operator!=(const viterator& other)
{
return t!=other.t;
}
inline bool operator!=(const viterator& other)const
{
return t!=other.t;
}
inline bool operator==(const viterator& other)
{
return t==other.t;
}
inline bool operator==(const viterator& other)const
{
return t==other.t;
}
inline bool operator<(const viterator& other)
{
return t<other.t;
}
inline bool operator<(const viterator& other)const
{
return t<other.t;
}
inline bool operator<=(const viterator& other)
{
return t<=other.t;
}
inline bool operator<=(const viterator& other)const
{
return t<=other.t;
}
inline bool operator>(const viterator& other)
{
return t>other.t;
}
inline bool operator>(const viterator& other)const
{
return t>other.t;
}
inline bool operator>=(const viterator& other)
{
return t>=other.t;
}
inline bool operator>=(const viterator& other)const
{
return t>=other.t;
}
private:
T* t;
};
template
class cvector:public NoCopy
{
public:
typedef viterator iterator;//viterator就是对一个指针的包装,所以完
全可以用T代替viterator
typedef const viterator const_iterator;
//typedef T iterator;
//typedef const T* const_iterator;
cvector()
{
initData(0);
}
cvector(int capa,const T& val=T())
{
initData(capa);
newCapacity(capacity_);
for (int i=0;i<size_;i++)
buf[i]=val;
}
cvector(const_iterator first,const_iterator last)
{
initData(last-first);
newCapacity(capacity_);
iterator iter=iterator(first);
int index=0;
while(iter!=last)
buf[index++]=*iter++;
}
~cvector()
{
if (buf)
{
delete[] buf;
buf=NULL;
}
size_=capacity_=0;
}
void clear()
{
if (buf)
erase(begin(),end());
}
void push_back(const T& t)
{
if (size_==capacity_)
{
int capa=calculateCapacity();
newCapacity(capa);
}
buf[size_++]=t;
}
void pop_back()
{
if (!empty())
erase(end() - 1);
}
int insert(const_iterator iter,const T& t )
{
int index=iter-begin();
if (index<=size_)
{
if (size_==capacity_)
{
int capa=calculateCapacity();
newCapacity(capa);
}
memmove(buf+index+1,buf+index,(size_-index)*sizeof(T));
buf[index]=t;
size_++;
}
return index;
}
iterator erase(const_iterator iter)
{
int index=iter-begin();
if (index<size_ && size_>0)
{
memmove(buf+index ,buf+index+1,(size_-index)*sizeof(T));
buf[–size_]=T();
}
return iterator(iter);
}
iterator erase(const_iterator first,const_iterator last)
{
iterator first_=iterator(first);
iterator last_=iterator(last);
while(first_<=last_–)
erase(first_);
return iterator(first_);
} T& front()
{
assert(size_>0);
return buf[0];
}
T& back()
{
assert(size_>0);
return buf[size_-1];
}
T& at(int index)
{
assert(size_>0);
return buf[index];
}
T& operator[](int index)
{
assert(size_>0 && index>=0 && index<size_);
return buf[index];
}
bool empty() const
{
return size_==0;
}
int size() const
{
return size_;
}
int capacity() const
{
return capacity_;
}
iterator begin()
{
return iterator(&buf[0]);
}
iterator end()
{
return iterator(&buf[size_]);
}
private:
void newCapacity(int capa)
{
capacity_=capa;
T* newBuf=new T[capacity_];
if (buf)
{
memcpy(newBuf,buf,size_*sizeof(T));
delete [] buf;
}
buf=newBuf;
}
inline int calculateCapacity()
{ return capacity_*3/2+1 ;
}
inline void initData(int capa)
{
buf=NULL;
size_=capacity_=capa>0?capa:0;
}
int size_;
int capacity_ ;
T* buf;
};
struct Point
{
Point(int x_=0,int y_=0):x(x_),y(y_){}
int x,y;
};
bool operator<(const Point& p1,const Point& p2)
{
if(p1.x<p2.x)
{
return true;
}else if(p1.x>p2.x)
{
return false;
}
return p1.y<p2.y;
}
void cvectorTest()
{
cvector vect;
for (int i=0;i<10;i++)
{
Point p(i,i);
vect.push_back§;
}
cvector::iterator iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
iter=vect.begin()+5;
vect.insert(iter,Point(55,55));
iter=vect.end()-3;
vect.insert(iter,Point(77,77));
cout<<endl<<endl<<“插入两个元素后:”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
std::sort(vect.begin(),vect.end());
cout<<endl<<endl<<“排序后:”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
vect.erase(vect.begin()+10);
vect.erase(vect.begin()+10);
cout<<endl<<endl<<“删除之前新增的两个元素”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
vect.clear();
cout<<endl<<endl<<“执行clear之后”<<endl;
cout<<“size=”<<vect.size()<<“,capacity=”<<vect.capacity();
cvector vect1;
for (int i=10;i<20;i++)
{
Point p(i,i);
vect1.push_back§;
}
cout<<endl<<endl<<“从别的cvector复制数据:”<<endl;
cvector vect2(vect1.begin(),vect1.end());
vect2.pop_back();
vect2.pop_back();
for(int i=0;i<vect2.size();i++)
{
cout<<“[”<<vect2[i].x<<“,”<<vect2[i].y<<"], ";
}
cout<<endl;
}
}
//实例代码级运行结果
struct Point
{
Point(int x_=0,int y_=0):x(x_),y(y_){}
int x,y;
};
bool operator<(const Point& p1,const Point& p2)
{
if(p1.x<p2.x)
{
return true;
}else if(p1.x>p2.x)
{
return false;
}
return p1.y<p2.y;
}
void cvectorTest()
{
cvector vect;
for (int i=0;i<10;i++)
{
Point p(i,i);
vect.push_back§;
}
cvector::iterator iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
iter=vect.begin()+5;
vect.insert(iter,Point(55,55));
iter=vect.end()-3;
vect.insert(iter,Point(77,77));
cout<<endl<<endl<<“插入两个元素后:”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
std::sort(vect.begin(),vect.end());
cout<<endl<<endl<<“排序后:”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
vect.erase(vect.begin()+10);
vect.erase(vect.begin()+10);
cout<<endl<<endl<<“删除之前新增的两个元素”<<endl;
iter=vect.begin();
while (iter!=vect.end())
{
cout<< “[” << iter->x << " " << iter->y <<"], ";
++iter;
}
vect.clear();
cout<<endl<<endl<<“执行clear之后”<<endl;
cout<<“size=”<<vect.size()<<“,capacity=”<<vect.capacity();
cvector vect1;
for (int i=10;i<20;i++)
{
Point p(i,i);
vect1.push_back§;
}
cout<<endl<<endl<<“从别的cvector复制数据:”<<endl;
cvector vect2(vect1.begin(),vect1.end());
vect2.pop_back();
vect2.pop_back();
for(int i=0;i<vect2.size();i++)
{
cout<<“[”<<vect2[i].x<<“,”<<vect2[i].y<<"], ";
}
cout<<endl;
}
1.4.14 简述 STL 中的 map 的实现原理
map是关联式容器,它们的底层容器都是红黑树。map 的所有元素都是 pair,同时拥有实值
(value)和键值(key)。pair 的第一元素被视为键值,第二元素被视为实值。所有元素都会根据元素
的键值自动被排序。不允许键值重复。
map的特性如下
(1)map以RBTree作为底层容器;
(2)所有元素都是 键+值 存在;
(3)不允许 键 重复;
(4)所有元素 是通过 键 进行自动排序的;
(5)map的 键 是不能修改的,但是其键对应的值是可以修改的。
1.4.15 C++ 的 vector 和 list中,如果删除末尾的元素,其指针和迭代器如何变
化?若删除的是中间的元素呢?
迭代器和指针之间的区别
迭代器不是指针,是类模板,表现的像指针。他只是模拟了指针的一些功能,重载了指针的一
些操作符,–>、++、–等。迭代器封装了指针,是一个”可遍历STL( Standard Template
Library)容器内全部或部分元素”的对象,本质是封装了原生指针,是指针概念的一种提升,提供
了比指针更高级的行为,相当于一种智能指针,他可以根据不同类型的数据结构来实现不同的
++,–等操作。
迭代器返回的是对象引用 而不是对象的值,所以cout只能输出迭代器使用取值后的值而不能直
接输出其自身。
vector和list特性
vector特性
动态数组。元素在内存连续存放。随机存取任何元素都在常数时间完成。在尾端增删元素具有较大的性能(大部分情况下是常数时间)。
list特性 双向链表。元素在内存不连续存放。在任何位置 增删元素都能在常数时间完成。不支持
随机存取。
vector增删元素
对于vector而言,删除某个元素以后,该元素后边的每个元素的迭代器都会失效,后边每个元素都
往前移动一位,erase返回下一个有效的迭代器。
list增删元素
对于list而言,删除某个元素,只有“指向被删除元素”的那个迭代器失效,其它迭代器不受任何影
响。
1.4.16 请你来说一下 map 和 set 有什么区别,分别又是怎么实现的?
参考回答
set是一种关联式容器,其特性如下:
(1)set以RBTree作为底层容器
(2)所得元素的只有key没有value,value就是key
(3)不允许出现 键值 重复
(4)所有的元素都会被自动排序
(5)不能通过迭代器来改变set的值,因为set的值就是键,set的迭代器是const的
map和set一样是关联式容器,其特性如下:
(1)map以RBTree作为底层容器
(2)所有元素都是键+值存在
(3)不允许 键重复
(4)所有元素是通过键进行自动排序的
(5)map的键是不能修改的,但是其键对应的值是可以修改的
综上所述,map和set底层实现都是红黑树;map和set的区别在于map的值不作为键,键和值是
分开的。
1.4.17 说说 push_back 和 emplace_back 的区别
参考回答
如果要将一个临时变量push到容器的末尾,push_back()需要先构造临时对象,再将这个对象拷贝到
容器的末尾, 而emplace_back()则直接在容器的末尾构造对象,这样就省去了拷贝的过程。
答案解析
参考代码:
#include
#include
#include
using namespace std;
class A {
public:
A(int i){
str = to_string(i);
cout << “构造函数” << endl;
}
~A(){}
A(const A& other): str(other.str){
cout << “拷贝构造” << endl;
}public:
string str;
};
int main()
{
vector vec;
vec.reserve(10);
for(int i=0;i<10;i++){
vec.push_back(A(i)); //调用了10次构造函数和10次拷贝构造函数,
// vec.emplace_back(i);
//调用了10次构造函数一次拷贝构造函数都没有调用过
}
}
1.4.18 STL 中 vector 与 list 具体是怎么实现的?常见操作的时间复杂度是多少?
参考回答
vector 一维数组(元素在内存连续存放)
是动态数组,在堆中分配内存,元素连续存放,有保留内存,如果减少大小后,内存也不会释
放;如果新增大小 大于 当前大小时才会重新分配内存。
扩容方式: a. 倍放开辟三倍的内存
b. 旧的数据开辟到新的内存
c. 释放旧的内存
d. 指向新内存
list 双向链表(元素存放在堆中)
元素存放在堆中,每个元素都是放在一块内存中,它的内存空间可以是不连续的,通过指针来
进行数据的访问,这个特点,使得它的随机存取变得非常没有效率,因此它没有提供[ ]操作符的重
载。但是由于链表的特点,它可以很有效的支持任意地方的删除和插入操作。
特点:a. 随机访问不方便
b. 删除插入操作方便
常见时间复杂度
(1)vector插入、查找、删除时间复杂度分别为:O(n)、O(1)、O(n);
(2)list插入、查找、删除时间复杂度分别为:O(1)、O(n)、O(1)。
1.5 新特性
1.5.1 说说 C++11 的新特性有哪些
参考回答
C++新特性主要包括包含 语法改进 和标准库 扩充两个方面,主要包括以下11点:
语法的改进
(1)统一的初始化方法
(2)成员变量默认初始化
(3)auto关键字 用于定义变量,编译器可以自动判断的类型(前提:定义一个变量时对其进行
初始化)
(4)decltype 求表达式的类型
(5)智能指针 shared_ptr
(6)空指针 nullptr(原来NULL)
(7)基于范围的for循环
(8)右值引用和move语义 让程序员有意识减少进行深拷贝操作
标准库扩充(往STL里新加进一些模板类,比较好用)
(9)无序容器(哈希表) 用法和功能同map一模一样,区别在于哈希表的效率更高
(10)正则表达式 可以认为正则表达式实质上是一个字符串,该字符串描述了一种特定模式的字
符串
(11)Lambda表达式
答案解析
统一的初始化方法
C++98/03 可以使用初始化列表(initializer list)进行初始化:
int i_arr[3] = { 1, 2, 3 };
long l_arr[] = { 1, 3, 2, 4 };
struct A
{
int x;
int y;
} a = { 1, 2 };
** 但是这种初始化方式的适用性非常狭窄,只有上面提到的这两种数据类型可以使用初始化列
表。在 C++11 中,初始化列表的适用性被大大增加了。它现在可以用于任何类型对象的初始化**,
实例如下:
class Foo
{
public:
Foo(int) {}
private:
Foo(const Foo &);
};
int main(void)
{
Foo a1(123);
Foo a2 = 123; //error: ‘Foo::Foo(const Foo &)’ is private
Foo a3 = { 123 };
Foo a4 { 123 };
int a5 = { 3 };
int a6 { 3 };
return 0;
}
在上例中,a3、a4 使用了新的初始化方式来初始化对象,效果如同 a1 的直接初始化。a5、a6
则是基本数据类型的列表初始化方式。可以看到,它们的形式都是统一的。这里需要注意的是,
a3 虽然使用了等于号,但它仍然是列表初始化,因此,私有的拷贝构造并不会影响到它。a4 和
a6 的写法,是 C++98/03 所不具备的。在 C++11 中,可以直接在变量名后面跟上初始化列表,来
进行对象的初始化。
成员变量默认初始化
好处:构建一个类的对象不需要用构造函数初始化成员变量。
//程序实例
#include
using namespace std;
class B
{
public:
int m = 1234; //成员变量有一个初始值
int n;
};
int main()
{
B b;
cout << b.m << endl;
return 0;
}
auto关键字
用于定义变量,编译器可以自动判断的类型(前提:定义一个变量时对其进行初始化)
/程序实例
#include
using namespace std;
int main(){
vector< vector > v;
vector< vector >::iterator i = v.begin();
return 0;
}
可以看出来,定义迭代器 i 的时候,类型书写比较冗长,容易出错。然而有了 auto 类型推导,
我们大可不必这样,只写一个 auto 即可。
decltype 求表达式的类型
decltype 是 C++11 新增的一个关键字,它和 auto 的功能一样,都用来在编译时期进行自动类
型推导。
(1)为什么要有decltype
因为 auto 并不适用于所有的自动类型推导场景,在某些特殊情况下 auto 用起来非常不方便,
甚至压根无法使用,所以 decltype 关键字也被引入到 C++11 中。
auto 和 decltype 关键字都可以自动推导出变量的类型,但它们的用法是有区别的:
auto varname = value;
decltype(exp) varname = value;
其中,varname 表示变量名,value 表示赋给变量的值,exp 表示一个表达式。
auto 根据"=“右边的初始值 value 推导出变量的类型,而 decltype 根据 exp 表达式推导出变量的类型,跟”="右边的 value 没有关系。
另外,auto 要求变量必须初始化,而 decltype 不要求。这很容易理解,auto 是根据变量的初
始值来推导出变量类型的, 如果不初始化,变量的类型也就无法推导了。decltype 可以写成下面
的形式:
decltype(exp) varname;
(2)代码示例
// decltype 用法举例
int a = 0;
decltype(a) b = 1; //b 被推导成了 int
decltype(10.8) x = 5.5; //x 被推导成了 double
decltype(x + 100) y; //y 被推导成了 double
智能指针 shared_ptr
和 unique_ptr、weak_ptr 不同之处在于, 多个 shared_ptr 智能指针可以共同使用同一块堆内
存。并且,由于该类型智能指针在实现上采用的是引用计数机制,即便有一个 shared_ptr 指针放
弃了堆内存的“使用权”(引用计数减 1),也不会影响其他指向同一堆内存的 shared_ptr 指针
(只有引用计数为 0 时,堆内存才会被自动释放)。
#include
#include
using namespace std;
int main()
{
//构建 2 个智能指针
std::shared_ptr p1(new int(10));
std::shared_ptr p2(p1);
//输出 p2 指向的数据
cout << *p2 << endl;
p1.reset();//引用计数减 1,p1为空指针
if (p1) {
cout << “p1 不为空” << endl;
}
else {
cout << “p1 为空” << endl;
}
//以上操作,并不会影响 p2
cout << p2 << endl;
//判断当前和 p2 同指向的智能指针有多少个
cout << p2.use_count() << endl;
return 0;
}
/
程序运行结果:
10
p1 为空
10
1
*/
6.空指针 nullptr(原来NULL)
nullptr 是 nullptr_t 类型的右值常量,专用于初始化空类型指针。nullptr_t 是 C++11 新增加的
数据类型,可称为“指针空值类型”。也就是说,nullpter 仅是该类型的一个实例对象(已经定义
好,可以直接使用), 如果需要我们完全定义出多个同 nullptr 完全一样的实例对象。值得一提的
是,nullptr 可以被隐式转换成任意的指针类型。例如:
int * a1 = nullptr;
char * a2 = nullptr;
double * a3 = nullptr;
显然,不同类型的指针变量都可以使用 nullptr 来初始化,编译器分别将 nullptr 隐式转换成
int 、 char 以及 double* 指针类型。另外,通过将指针初始化为 nullptr,可以很好地解决 NULL
遗留的问题,比如:
#include
using namespace std;
void isnull(void c){
cout << "voidc" << endl;
}
void isnull(int n){
cout << “int n” << endl;
}
int main() {
isnull(NULL);
isnull(nullptr);
return 0;
}
/*
程序运行结果:
int n
void*c
*/
基于范围的for循环
如果要用 for 循环语句遍历一个数组或者容器,只能套用如下结构:
for(表达式 1; 表达式 2; 表达式 3){
//循环体
}
//程序实例
#include
#include
#include <string.h>
using namespace std;
int main() {
char arc[] = “www.123”;
int i; //for循环遍历普通数组
for (i = 0; i < strlen(arc); i++) {
cout << arc[i];
}
cout << endl;
vectormyvector(arc,arc+3);
vector::iterator iter;
//for循环遍历 vector 容器
for (iter = myvector.begin(); iter != myvector.end(); ++iter) {
cout << *iter;
}
return 0;
}
/*
程序运行结果:
www.123
www
*/
右值引用和move语义
右值引用]
C++98/03 标准中就有引用,使用 "&" 表示。但此种引用方式有一个缺陷,即正常情况下
只能操作 C++ 中的左值,无法对右值添加引用。举个例子:
int num = 10;
int &b = num; //正确
int &c = 10; //错误
如上所示,编译器允许我们为 num 左值建立一个引用,但不可以为 10 这个右值建立引
用。因此,C++98/03 标准中的引用又称为左值引用。
注意,虽然 C++98/03 标准不支持为右值建立非常量左值引用,但允许使用常量左值引用操
作右值。也就是说,常量左值引用既可以操作左值,也可以操作右值,例如:
我们知道,右值往往是没有名称的,因此要使用它只能借助引用的方式。这就产生一个问
题,实际开发中我们可能需要对右值进行修改(实现移动语义时就需要),显然左值引用的
方式是行不通的。
为此,C++11 标准新引入了另一种引用方式,称为右值引用,用 “&&” 表示。
需要注意的,和声明左值引用一样,右值引用也必须立即进行初始化操作,且只能使用右
值进行初始化,比如:
int num = 10;
//int && a = num;
int && a = 10;
//右值引用不能初始化为左值
和常量左值引用不同的是,右值引用还可以对右值进行修改。例如:
int && a = 10;
a = 100;
cout << a << endl;
/*
程序运行结果:
100
*/
另外值得一提的是,C++ 语法上是支持定义常量右值引用的,例如:
const int&& a = 10;//编译器不会报错
但这种定义出来的右值引用 并无实际用处。一方面,右值引用主要用于移动语义和完美转
发,其中前者需要有修改右值的权限;其次,常量右值引用的作用就是引用一个不可修改的
右值,这项工作完全可以交给常量左值引用完成。
move语义
move 本意为 “移动”,但该 函数并不能移动 任何数据,它的功能很简单,就是将 某个左值
强制转化为右值。基于 move() 函数特殊的功能,其常用于实现移动语义。move() 函数的用
法也很简单,其语法格式如下:
move( arg ) //其中,arg 表示指定的左值对象。该函数会返回 arg 对象的右值形式。
//程序实例
#include
using namespace std;
class first {
public:
first() :num(new int(0)) {
cout << “construct!” << endl;
}
//移动构造函数
first(first &&d) :num(d.num) {
d.num = NULL;
cout << “first move construct!” << endl;
}
public: //这里应该是 private,使用 public 是为了更方便说明问题
int *num;
};
class second {
public:
second() :fir() {}
//用 first 类的移动构造函数初始化 fir
second(second && sec) :fir(move(sec.fir)) {
cout << “second move construct” << endl;
}
public: //这里也应该是 private,使用 public 是为了更方便说明问题
first fir;
};
int main() {
second oth; second oth2 = move(oth);
//cout << oth.fir.num << endl; //程序报运行时错误
return 0;
}
/
程序运行结果:
construct!
first move construct!
无序容器(哈希表)
用法和功能同map一模一样,区别在于哈希表的效率更高。
(1) 无序容器具有以下 2 个特点:
a. 无序容器内部存储的键值对是无序的,各 键值对的存储位置 取决于 该键值对中的 键,
b. 和 关联式容器相比,无序容器 擅长通过 指定键 查找对应的值(平均时间复杂度为 O(1));但
对于 使用迭代器遍历容器中存储的元素,无序容器的执行效率则不如关联式容器。
(2) 和 关联式容器一样, 无序容器只是一类容器的统称,其包含有 4 个具体容器,分别为
unordered_map、unordered_multimap、unordered_set 以及 unordered_multiset。功能如下
表:
image-20220711213744512
(3) 程序实例(以 unordered_map 容器为例)
#include
#include
#include <unordered_map>
using namespace std;
int main()
{
//创建并初始化一个 unordered_map 容器,其存储的 <string,string> 类型的键值对
std::unordered_map<std::string, std::string> my_uMap{
{“教程1”,“www.123”},
{“教程2”,“www.234”},
{“教程3”,“www.345”} }; //查找指定键对应的值,效率比关联式容器高
string str = my_uMap.at(“C语言教程”);
cout << "str = " << str << endl;
//使用迭代器遍历哈希容器,效率不如关联式容器
for (auto iter = my_uMap.begin(); iter != my_uMap.end(); ++iter)
{
//pair 类型键值对分为 2 部分
cout << iter->first << " " << iter->second << endl;
}
return 0;
}
/*
程序运行结果:
教程1 www.123
教程2 www.234
教程3 www.345
*/
正则表达式
可以认为正则表达式实质上是一个字符串,该字符串描述了一种特定模式的字符串。常用符号的意
义如下:
image-20220711214142879
image-20220711214204421
Lambda匿名函数
所谓匿名函数,简单地理解就是没有名称的函数,又常被称为 lambda 函数或者 lambda 表达
式。
(1)定义
lambda 匿名函数很简单,可以套用如下的语法格式:
[外部变量访问方式说明符] (参数) mutable noexcept/throw() -> 返回值类型
{
函数体;
};
其中各部分的含义分别为:
a. [外部变量方位方式说明符]
[ ] 方括号用于向编译器表明当前是一个 lambda 表达式,其不能被省略。在方括号内部,可以
注明当前 lambda 函数的函数体中可以使用哪些“外部变量”。
所谓外部变量,指的是和当前 lambda 表达式位于同一作用域内的所有局部变量。
b. (参数)
和普通函数的定义一样,lambda 匿名函数也可以接收外部传递的多个参数。和普通函数不同的
是,如果不需要传递参数,可以连同 () 小括号一起省略;
c. mutable
此关键字可以省略,如果使用则之前的 () 小括号将不能省略(参数个数可以为 0)。默认情况
下,对于以值传递方式引入的外部变量,不允许在 lambda 表达式内部修改它们的值(可以理解
为这部分变量都是 const 常量)。而如果想修改它们,就必须使用 mutable 关键字。
注意: 对于以值传递方式引入的外部变量,lambda 表达式修改的是拷贝的那一份,并不会修改
真正的外部变量;
d. noexcept/throw()
可以省略,如果使用,在之前的 () 小括号将不能省略(参数个数可以为 0)。默认情况下,
lambda 函数的函数体中可以抛出任何类型的异常。而标注 noexcept 关键字,则表示函数体内不
会抛出任何异常;使用 throw() 可以指定 lambda 函数内部可以抛出的异常类型。
e. -> 返回值类型
指明 lambda 匿名函数的返回值类型。值得一提的是,如果 lambda 函数体内只有一个 return
语句,或者该函数返回 void,则编译器可以自行推断出返回值类型,此情况下可以直接省略"-> 返
回值类型"。
f. 函数体
和普通函数一样,lambda 匿名函数包含的内部代码都放置在函数体中。该函数体内除了可以使
用指定传递进来的参数之外,还可以使用指定的外部变量以及全局范围内的所有全局变量。
(2)程序实例
#include
#include
using namespace std;
int main()
{
int num[4] = {4, 2, 3, 1};
//对 a 数组中的元素进行排序
sort(num, num+4, [=](int x, int y) -> bool{ return x < y; } );
for(int n : num){
cout << n << " ";
}
return 0;
}
程序运行结果:
/*
1 2 3 4
*/
1.5.2 说说 C++ 中智能指针和指针的区别是什么?
参考回答
智能指针
如果在程序中使用new从堆(自由存储区)分配内存,等到不需要时,应使用delete将其释放。
C++引用了智能指针auto_ptr,以帮助自动完成这个过程。随后的编程体验(尤其是使用STL)表
明,需要有更精致的机制。基于程序员的编程体验和BOOST库提供的解决方案, C++11摒弃了
auto_ptr, 并新增了三种智能指针:unique_ptr、shared_ptr和weak_ptr。所有新增的智能指针
都能与STL容器和移动语义协同工作。
指针
C 语言规定 所有变量在使用前必须先定义,指定其类型,并按此分配内存单元。指针变量 不同于
整型变量和其他类型的变量,它是专门用来存放地址的,所以必须将它定义为“指针类型”。
智能指针和普通指针的区别
智能指针和普通指针的区别在于 智能指针 实际上是对 普通指针 加了一层封装机制, 区别是它负责自动
释放所指的对象,这样的一层封装机制的目的是为了使得智能指针可以方便的管理一个对象的生命期。
1.5.3 说说 C++中的智能指针有哪些?分别解决的问题以及区别?
参考回答
C++中的智能指针有4种,分别为:shared_ptr、unique_ptr、weak_ptr、auto_ptr,其中
auto_ptr被C++11弃用。
使用智能指针的原因
申请的空间(即new出来的空间),在使用结束时,需要delete掉,否则会形成内存碎片。在程
序运行期间,new出来的对象,在析构函数中delete掉,但是这种方法不能解决所有问题, 因为有
时候new发生在某个全局函数里面, 该方法会给程序员造成精神负担。此时,智能指针就派上了用
场。使用智能指针可以很大程度上避免这个问题,因为智能指针就是一个类,当超出了类的作用域
时,类会自动调用析构函数,析构函数会自动释放资源。所以,智能指针的作用原理就是在函数结
束时自动释放内存空间,避免了手动释放内存空间。
四种指针分别解决的问题以及各自特性如下:
(1)auto_ptr(C++98的方案,C++11已经弃用)
采用所有权模式。
auto_ptr p1(new string(“I reigned loney as a cloud.”));
auto_ptr p2;
p2=p1; //auto_ptr不会报错
此时不会报错,p2剥夺了p1的所有权,但是当程序运行时访问p1将会报错。所以auto_ptr的缺
点是:存在潜在的内存崩溃问题。
(2)unique_ptr(替换auto_ptr)
unique_ptr实现独占式拥有或严格拥有概念, 保证同一时间内只有一个智能指针可以指向该对
象。它对于避免资源泄露,例如,以new创建对象后因为发生异常而忘记调用delete时的情形特别
有用。
采用所有权模式,和上面例子一样。
auto_ptr p3(new string(“I reigned loney as a cloud.”));
auto_ptr p4;
p4=p3; //此时不会报错
编译器认为P4=P3非法,避免了p3不再指向有效数据的问题。因此,unique_ptr比auto_ptr更
安全。 另外unique_ptr还有更聪明的地方 :当程序试图将一个 unique_ptr 赋值给另一个时,如果
源 unique_ptr 是个临时右值,编译器允许这么做; 如果源 unique_ptr 将存在一段时间,编译器将禁止这么做,比如:
unique_ptr pu1(new string (“hello world”));
unique_ptr pu2;
pu2 = pu1; // #1 not allowed
unique_ptr pu3;
pu3 = unique_ptr(new string (“You”)); // #2 allowed
其中#1留下悬挂的unique_ptr(pu1),这可能导致危害。而#2不会留下悬挂的unique_ptr,因
为它调用 unique_ptr 的构造函数,该构造函数创建的临时对象在其所有权让给 pu3 后就会被销
毁。这种随情况而已的行为表明,unique_ptr 优于允许两种赋值的auto_ptr 。
注意:如果确实想执行类似与#1的操作,要安全的重用这种指针,可给它赋新值。C++有一个
标准库函数std::move(),让你能够将一个unique_ptr赋给另一个。例如:
unique_ptr ps1, ps2;
ps1 = demo(“hello”);
ps2 = move(ps1);
ps1 = demo(“alexia”);
cout << *ps2 << *ps1 << endl;
(3)shared_ptr(非常好使)
shared_ptr实现共享式拥有概念。多个智能指针可以指向相同对象,该对象和其相关资源会在
“最后一个引用被销毁”时候释放。从名字share就可以看出了资源可以被多个指针共享,它使用计
数机制来表明资源被几个指针共享。可以通过成员函数use_count()来查看资源的所有者个数。除
了可以通过new来构造,还可以通过传入auto_ptr, unique_ptr,weak_ptr来构造。当我们调用
release()时,当前指针会释放资源所有权,计数减一。当计数等于0时,资源会被释放。
shared_ptr 是为了解决 auto_ptr 在对象所有权上的局限性(auto_ptr 是独占的), 在使用引用计
数的机制上提供了可以共享所有权的智能指针。
成员函数:
use_count 返回引用计数的个数
unique 返回是否是独占所有权( use_count 为 1)
swap 交换两个 shared_ptr 对象(即交换所拥有的对象)
reset 放弃内部对象的所有权或拥有对象的变更, 会引起原有对象的引用计数的减少
get 返回内部对象(指针), 由于已经重载了()方法, 因此和直接使用对象是一样的.如 shared_ptr
sp(new int(1)); sp 与 sp.get()是等价的
(4)weak_ptr
weak_ptr 是一种不控制对象生命周期的智能指针, 它指向一个 shared_ptr 管理的对象。进行该
对象的内存管理的是那个强引用的 shared_ptr。weak_ptr只是提供了对管理对象的一个访问手
段。weak_ptr 设计的目的是为配合 shared_ptr 而引入的一种智能指针来协助 shared_ptr 工作,
它 只可以从一个 shared_ptr 或 另一个 weak_ptr 对象构造, 它的构造和析构不会引起引用记数的增
加或减少。weak_ptr是用来解决shared_ptr相互引用时的死锁问题,如果说两个shared_ptr相互
引用,那么这两个指针的引用计数永远不可能下降为0,资源永远不会释放。它是对对象的一种弱引
用,不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr可以直接赋值给它,它可以通过调用lock函数来获得shared_ptr。
class B;
class A
{
public:
shared_ptr pb_;
~A()
{
cout<<“A delete\n”;
}
};
class B
{
public:
shared_ptr pa_;
~B()
{
cout<<“B delete\n”;
}
};
void fun()
{
shared_ptr pb(new B());
shared_ptr pa(new A());
pb->pa_ = pa;
pa->pb_ = pb;
cout<<pb.use_count()<<endl;
cout<<pa.use_count()<<endl;
}
int main()
{
fun();
return 0;
}
可以看到fun函数中pa ,pb之间互相引用,两个资源的引用计数为2,当要跳出函数时,智能指
针pa ,pb析构时两个资源引用计数会减一,但是两者引用计数还是为1,导致跳出函数时资源没有
被释放(A B的析构函数没有被调用), 如果把其中一个改为weak_ptr就可以了,我们把类A里面
的shared_ptr pb_; 改为weak_ptr pb; 运行结果如下,这样的话,资源B的引用开始就只有1,当
pb析构时,B的计数变为0,B得到释放,B释放的同时也会使A的计数减一,同时pa析构时使A的
计数减一,那么A的计数为0,A得到释放。
注意:我们不能通过weak_ptr直接访问对象的方法,比如B对象中有一个方法print(),我们不能
这样访问,pa->pb->print(); 英文 pb是一个weak_ptr,应该先把它转化为shared_ptr,如:
shared_ptr p = pa->pb_.lock(); p->print();
1.5.4 简述 C++ 右值引用与转移语义
参考回答
右值引用
一般来说,不能取地址的表达式,就是右值引用,能取地址的,就是左值。
class A { };
A & r = A(); //error,A()是无名变量,是右值
A && r = A(); //ok,r是右值引用
转移语义
move 本意为 “移动”,但该函数并不能移动任何数据,它的功能很简单,就是将某个左值强制转
化为右值。基于 move() 函数特殊的功能,其常用于实现移动语义。
右值引用
C++98/03 标准中就有引用,使用 “&” 表示。但此种引用方式有一个缺陷,即正常情况下只能操作
C++ 中的左值,无法对右值添加引用。举个例子:
int num = 10;
int &b = num; //正确
int &c = 10; //错误
如上所示,编译器允许我们为 num 左值建立一个引用,但不可以为 10 这个右值建立引用。因此,
C++98/03 标准中的引用又称为左值引用。
注意:虽然 C++98/03 标准不支持 为右值建立非常量左值引用,但允许使用常量左值引用 操作右值。
也就是说,常量左值引用既可以操作左值,也可以操作右值,例如:
int num = 10;
const int &b = num;
const int &c = 10;
我们知道,右值往往是没有名称的,因此要使用它只能借助引用的方式。这就产生一个问题,实际开
发中我们可能需要对右值进行修改(实现移动语义时就需要),显然左值引用的方式是行不通的。
为此,C++11 标准新引入了另一种引用方式,称为右值引用,用 “&&” 表示。
注意:和声明左值引用一样,右值引用也必须立即进行初始化操作,且只能使用右值进行初始化,比
如:
int num = 10;
//int && a = num;
//右值引用不能初始化为左值
int && a = 10;
和常量左值引用不同的是,右值引用还可以对右值进行修改。例如:
int && a = 10;
a = 100;
cout << a << endl;
/*
程序运行结果:
100
*/
另外值得一提的是,C++ 语法上是支持定义常量右值引用的,例如:
const int&& a = 10;//编译器不会报错
但这种定义出来的右值引用并无实际用处。一方面,右值引用主要用于移动语义和完美转发,其中前
者需要有修改右值的权限;其次,常量右值引用的作用就是引用一个不可修改的右值,这项工作完全可
以交给常量左值引用完成。
move语义
//程序实例
#include
using namespace std;
class first {
public:
first() :num(new int(0)) {
cout << “construct!” << endl;
}
//移动构造函数
first(first &&d) :num(d.num) {
d.num = NULL;
cout << “first move construct!” << endl;
}
public: //这里应该是 private,使用 public 是为了更方便说明问题
int *num;
};
class second {
public:
second() :fir() {}
//用 first 类的移动构造函数初始化 fir
second(second && sec) :fir(move(sec.fir)) {
cout << “second move construct” << endl;
}
public: //这里也应该是 private,使用 public 是为了更方便说明问题
first fir;
};
int main() {
second oth;
second oth2 = move(oth);
//cout << oth.fir.num << endl; //程序报运行时错误
return 0;
}
/
程序运行结果:
construct!
first move construct!
second move construct
*/
1.5.5 简述 C++ 中智能指针的特点
参考回答
C++中的智能指针有4种,分别为:shared_ptr、unique_ptr、weak_ptr、auto_ptr,其中
auto_ptr被C++11弃用。
为什么要使用智能指针: 智能指针的作用是 管理一个指针,因为存在 申请的空间在 函数结束时忘记
释放,造成内存泄漏的情况。使用智能指针可以很大程度上避免这个问题,因为智能指针就是一个*
类,当超出了类的作用域时,类会自动调用析构函数,自动释放资源。
四种指针各自特性
( 1)auto_ptr
auto指针存在的问题是,两个智能指针同时指向一块内存,就会两次释放同一块资源,自然报
错。
(2)unique_ptr
unique指针规定 一个智能指针 独占一块内存资源。当两个智能指针同时指向一块内存,编译报
错
实现原理: 将 拷贝构造函数和赋值拷贝构造函数 申明为private或delete。不允许拷贝构造函数
和赋值操作符, 但是支持移动构造函数,通过std:move把一个对象指针变成右值之后可以移动给
另一个unique_ptr
(3)shared_ptr
共享指针可以实现多个智能指针指向相同对象,该对象和其相关资源会在引用为0时被销毁释
放。
实现原理:有一个引用计数的指针类型变量,专门用于引用计数,使用拷贝构造函数和赋值拷
贝构造函数时,引用计数加1,当引用计数为0时,释放资源。
注意:weak_ptr、shared_ptr存在一个问题,当两个shared_ptr指针相互引用时,那么这两个指针的引
用计数不会下降为0,资源得不到释放。因此引入weak_ptr,weak_ptr是弱引用,weak_ptr的构造和析
构不会引起引用计数的增加或减少。
1.5.6 weak_ptr 能不能知道对象计数为 0,为什么?
参考回答
不能。
weak_ptr是一种不控制 对象生命周期的智能指针,它指向一个shared_ptr管理的对象。进行 该对象
管理的是 那个引用的shared_ptr。weak_ptr只是提供了对管理 对象的一个访问手段。weak_ptr设计的
目的只是为了配合shared_ptr而引入的一种智能指针,配合shared_ptr工作,它只可以从一个
shared_ptr或者另一个weak_ptr对象构造 ,它的构造和析构不会引起计数的增加或减少。
1.5.7 weak_ptr 如何解决 shared_ptr 的循环引用问题?
参考回答
为了解决循环引用导致的内存泄漏,引入了弱指针weak_ptr, weak_ptr的构造函数不会修改引用计
数的值,从而不会对对象的内存进行管理,其类似一个普通指针,但是 不会指向 引用计数的共享内存,
但是可以检测到所管理的对象是否已经被释放,从而避免非法访问。
1.5.8 share_ptr 怎么知道跟它共享对象的指针释放了
参考回答
多个shared_ptr对象可以同时托管一个指针,系统会维护一个托管计数。当无shared_ptr托管该指针
时,delete该指针。
1.5.9 说说智能指针及其实现,shared_ptr 线程安全性,原理
参考回答
C++里面的四个智能指针: auto_ptr, shared_ptr, weak_ptr, unique_ptr 其中后三个是c++11支
持,并且第一个已经被11弃用。
为什么要使用智能指针
智能指针的作用是管理一个指针, 因为存在以下这种情况:
申请的空间在函数结束时忘记释放,造成内存泄漏。使用智能指针可以很大程度上的避免这个问题,因为智能指针就是一个类,当
超出了类的作用域是,类会自动调用析构函数,析构函数会自动释放资源。
所以智能指针的作用原理 就是在函数结束时自动释放内存空间,不需要手动释放内存空间。
四种指针分别解决的问题以及各自特性如下:
(1)auto_ptr(C++98的方案,C++11已经弃用)
//程序实例
auto_ptr p1(new string(“I reigned loney as a cloud.”));
auto_ptr p2;
p2=p1; //auto_ptr不会报错
采用所有权模式。此时不会报错,p2剥夺了p1的所有权,但是当程序运行时访问p1将会报错。
所以auto_ptr的缺点是:存在潜在的内存崩溃问题。
(2)unique_ptr(替换auto_ptr)
nique_ptr实现独占式拥有或严格拥有概念,保证同一时间内只有一个智能指针可以指向该对
象。它对于避免资源泄露,例如,以new创建对象后因为发生异常而忘记调用delete时的情形特
别有用
//程序实例
auto_ptr p3(new string(“I reigned loney as a cloud.”));
auto_ptr p4;
p4=p3; //此时不会报错
采用所有权模式,和上面例子一样。编译器认为 P4=P3非法,避免了p3不再指向有效数据的问
题。因此,unique_ptr比auto_ptr更安全。 另外unique_ptr还有更聪明的地方:当程序试图将一
个 unique_ptr 赋值给另一个时,如果源 unique_ptr 是个临时右值,编译器允许这么做;如果源
unique_ptr 将存在一段时间,编译器将禁止这么做,比如以下代码
//程序实例
unique_ptr pu1(new string (“hello world”));
unique_ptr pu2;
pu2 = pu1; // #1 not allowed
unique_ptr pu3;
pu3 = unique_ptr(new string (“You”)); // #2 allowed
其中#1留下悬挂的unique_ptr(pu1),这可能导致危害。而#2不会留下悬挂的unique_ptr,因
为它调用 unique_ptr 的构造函数,该构造函数创建的临时对象在其所有权让给 pu3 后就会被销
毁。这种随情况而已的行为表明,unique_ptr 优于允许两种赋值的auto_ptr 。
注意:如果确实想执行类似与#1的操作,要安全的重用这种指针,可给它赋新值。C++有一个
标准库函数std::move(),让你能够将一个unique_ptr赋给另一个。例如:
//程序实例
unique_ptr ps1, ps2;
ps1 = demo(“hello”);
ps2 = move(ps1);
ps1 = demo(“alexia”);
cout << *ps2 << *ps1 << endl;
(3)shared_ptr(非常好使)
shared_ptr实现共享式拥有概念。多个智能指针可以指向相同对象,该对象和其相关资源会在“最后
一个引用被销毁”时候释放。从名字share就可以看出了资源可以被多个指针共享,它使用计数机制来表
明资源被几个指针共享。可以通过成员函数use_count()来查看资源的所有者个数。除了可以通过new
来构造,还可以通过传入auto_ptr, unique_ptr,weak_ptr来构造。当我们调用release()时,当前指针会
释放资源所有权,计数减一。当计数等于0时,资源会被释放。
shared_ptr 是为了解决 auto_ptr 在对象所有权上的局限性(auto_ptr 是独占的), 在使用引用计数的机
制上提供了可以共享所有权的智能指针。其成员函数如下:
use_count 返回引用计数的个数
unique 返回是否是独占所有权( use_count 为 1)
swap 交换两个 shared_ptr 对象(即交换所拥有的对象)
reset 放弃内部对象的所有权或拥有对象的变更, 会引起原有对象的引用计数的减少
get 返回内部对象(指针), 由于已经重载了()方法, 因此和直接使用对象是一样的.如 shared_ptr
sp(new int(1)); sp 与 sp.get()是等价的
(4)weak_ptr
weak_ptr 是一种不控制对象生命周期的智能指针, 它指向一个 shared_ptr 管理的对象。进行 该对象
的内存管理的 是那个强引用的 shared_ptr。weak_ptr只是提供了对管理对象的一个访问手段。
weak_ptr 设计的目的是为配合 shared_ptr 而引入的一种智能指针来协助 shared_ptr 工作,它只可
以从一个 shared_ptr 或另一个 weak_ptr 对象构造, 它的构造和析构不会引起引用记数的增加或减
少。
weak_ptr是用来解决shared_ptr相互引用时的死锁问题**,如果说两个shared_ptr相互引
用,那么这两个指针的引用计数永远不可能下降为0,资源永远不会释放。它是对对象的一种弱引用,
不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr 可以直接赋值给它, 它可以通
过调用lock函数来获得shared_ptr。
//程序实例
class B;
class A
{
public:
shared_ptr pb_;
~A()
{
cout<<“A delete\n”;
}
};
class B
{
public:
shared_ptr pa_;
~B()
{
cout<<“B delete\n”;
}
};
void fun()
{
shared_ptr pb(new B());
shared_ptr pa(new A());
pb->pa_ = pa;
pa->pb_ = pb;
cout<<pb.use_count()<<endl;
cout<<pa.use_count()<<endl;
}
int main()
{
fun();
return 0;
}
可以看到fun函数中pa ,pb之间互相引用,两个资源的引用计数为2,当要跳出函数时,智能指针
pa,pb析构时两个资源引用计数会减一,但是两者引用计数还是为1,导致跳出函数时资源没有被释放
(A B的析构函数没有被调用),如果把其中一个改为weak_ptr就可以了,我们把类A里面的
shared_ptr pb_; 改为weak_ptr pb; 运行结果如下,这样的话,资源B的引用开始就只有1,当pb析构
时,B的计数变为0,B得到释放,B释放的同时也会使A的计数减一,同时pa析构时使A的计数减一,那
么A的计数为0,A得到释放。
注意:我们不能通过weak_ptr直接访问对象的方法,比如B对象中有一个方法print(),我们不能这样
访问,pa->pb->print(); 英文 pb是一个weak_ptr,应该先把它转化为shared_ptr,如:shared_ptr p =
pa->pb_.lock(); p->print();
线程安全性
多线程环境下,调用不同shared_ptr实例的成员函数是不需要额外的同步手段的,即使这些
shared_ptr拥有的是同样的对象。但是如果多线程访问(有写操作)同一个shared_ptr,则需要
同步,否则就会有race condition 发生。也可以使用 shared_ptr overloads of atomic functions
来防止race condition的发生。
多个线程同时读 同一个shared_ptr对象是线程安全的,但是如果是 多个线程 对同一个
shared_ptr对象进行 读和写,则需要加锁。
多线程读写shared_ptr所指向的同一个对象,不管是相同的shared_ptr对象,还是不同的
shared_ptr对象,也需要加锁保护。例子如下:
//程序实例
shared_ptr global_instance = make_shared(0);
std::mutex g_i_mutex;
void thread_fcn()
{
//std::lock_guardstd::mutex lock(g_i_mutex);
//shared_ptr local = global_instance;
for(int i = 0; i < 100000000; i++)
{
*global_instance = *global_instance + 1;
//*local = *local + 1;
}}
int main(int argc, char** argv)
{
thread thread1(thread_fcn);
thread thread2(thread_fcn);
thread1.join();
thread2.join();
cout << "*global_instance is " << *global_instance << endl;
return 0;
}
在线程函数thread_fcn的for循环中,2个线程同时对global_instance进行加1的操作。这就是典
型的非线程安全的场景,最后的结果是未定的,运行结果如下:
*global_instance is 197240539
如果使用的是每个线程的局部shared_ptr对象local,因为这些local指向相同的对象,因此结果
也是未定的,运行结果如下:
*global_instance is 160285803
因此,这种情况下必须加锁,将thread_fcn中的第一行代码的注释去掉之后,不管是使用
global_instance,还是使用local,得到的结果都是:
*global_instance is 200000000
1.5.10 请你回答一下智能指针有没有内存泄露的情况
参考回答
智能指针有内存泄露的情况发生。
智能指针发生内存泄露的情况
当两个对象同时使用一个shared_ptr成员变量指向对方,会造成循环引用,使引用计数失效,
从而导致内存泄露。
智能指针的内存泄漏如何解决?
为了解决循环引用导致的内存泄漏,引入了弱指针weak_ptr,weak_ptr的构造函数不会修改引
用计数的值,从而不会对对象的内存进行管理,其类似一个普通指针,但是 不会指向 引用计数的共
享内存,但是可以检测到所管理的对象是否已经被释放,从而避免非法访问。
//程序实例
#include
#include
using namespace std;
class Child;
class Parent{
private:
std::shared_ptr ChildPtr;
public:
void setChild(std::shared_ptr child) {
this->ChildPtr = child;}
void doSomething() {
if (this->ChildPtr.use_count()) {
}
}
~Parent() {
}
};
class Child{
private:
std::shared_ptr ParentPtr;
public:
void setPartent(std::shared_ptr parent) {
this->ParentPtr = parent;
}
void doSomething() {
if (this->ParentPtr.use_count()) {
}
}
~Child() {
}
};
int main() {
std::weak_ptr wpp;
std::weak_ptr wpc;
{
std::shared_ptr p(new Parent);
std::shared_ptr c(new Child);
p->setChild©;
c->setPartent§;
wpp = p;
wpc = c;
std::cout << p.use_count() << std::endl;
std::cout << c.use_count() << std::endl;
}
std::cout << wpp.use_count() << std::endl;
std::cout << wpc.use_count() << std::endl;
return 0;
}
/*
程序运行结果:
2
2
1
1
*/
上述代码中,parent有一个shared_ptr类型的成员指向孩子,而child也有一个shared_ptr类型的成
员指向父亲。然后在创建孩子和父亲对象时也使用了智能指针c和p,随后将c和p分别又赋值给child的
智能指针成员parent和parent的智能指针成员child。从而形成了一个循环引用。
1.5.11 简述一下 C++11 中四种类型转换
C++中四种类型转换分别为const_cast、static_cast、dynamic_cast、reinterpret_cast,四种转
换功能分别如下:
const_cast
将const变量转为 非const
static_cast
最常用,可以用于各种隐式转换,比如非const转const,static_cast可以用于类向上转换,但向
下转换能成功但是不安全。
dynamic_cast
能用于含有虚函数的类转换,用于类向上和向下转换
向上转换: 指子类向基类转换。
向下转换: 指基类向子类转换。
这两种转换,子类包含父类,当父类转换成子类时可能出现非法内存访问的问题
dynamic_cast 通过判断 变量运行时类型和要转换的类型是否相同 来判断是否 能够进行向下转
换。dynamic_cast可以做类之间上下转换,转换的时候会进行类型检查,类型相等成功转换,类
型不等转换失败。运用RTTI技术,RTTI是”Runtime Type Information”的缩写,意思是运行时类型
信息,它提供了运行时确定对象类型的方法。在c++层面主要体现在dynamic_cast和typeid,vs中
虚函数表的- 1 位置存放了指向type_info的指针,对于存在虚函数的类型,dynamic_cast和typeid
都会去查询type_info。
reinterpret_cast
reinterpret_cast可以做任何类型的转换,不过 不对转换结果保证,容易出问题。
注意:为什么不用C的强制转换: C的强制转换表面上看起来功能 强大什么都能转,但是转换不够明
确,不能进行错误检查,容易出错。
1.5.12 简述一下 C++ 11 中 auto 的具体用法
参考回答
auto用于 定义变量,编译器可以自动判断变量的类型。auto主要有以下几种用法:
auto的基本使用方法
(1)基本使用语法如下
auto name = value; //name 是变量的名字,value 是变量的初始值
注意:auto 仅仅是一个占位符,在编译器期间它会被真正的类型所替代。或者说,C++ 中的变量必
须是有明确类型的,只是这个类型是由编译器自己推导出来的。
(2)程序实例如下
auto n = 10;
auto f = 12.8;
auto p = &n;
auto url = “www.123”;
a. 第 1 行中,10 是一个整数,默认是 int 类型,所以推导出变量 n 的类型是 int。
b. 第 2 行中,12.8 是一个小数,默认是 double 类型,所以推导出变量 f 的类型是 double。
c. 第 3 行中,&n 的结果是一个 int* 类型的指针,所以推导出变量 f 的类型是 int。
d. 第 4 行中,由双引号 “” 包围起来的字符串是 const char 类型,所以推导出变量 url 的类型
是 const char*,也即一个常量指针。
auto和 const 的结合使用
(1) auto 与 const 结合的用法
a. 当类型不为引用时,auto 的推导结果将 不保留表达式的 const 属性;
b. 当类型为引用时,auto 的推导结果将保留表达式的 const 属性
(2)程序实例如下
int x = 0;
const auto n = x; //n 为 const int ,auto 被推导为 int
auto f = n; //f 为 const int,auto 被推导为 int(const 属性被抛弃)
const auto &r1 = x; //r1 为 const int& 类型,auto 被推导为 int
auto &r2 = r1; //r1 为 const int& 类型,auto 被推导为 const int 类型
a. 第 2 行代码中,n 为 const int,auto 被推导为 int。
b. 第 3 行代码中,n 为 const int 类型,但是 auto 却被推导为 int 类型,这说明当 = 右边的表达
式带有 const 属性时,auto 不会使用 const 属性,而是直接推导出 non-const 类型。
c. 第 4 行代码中,auto 被推导为 int 类型,这个很容易理解,不再赘述。
d. 第 5 行代码中,r1 是 const int & 类型,auto 也被推导为 const int 类型,这说明当 const 和
引用结合时,auto 的推导将保留 表达式的 const 类型。
使用auto定义迭代器
在使用 stl 容器的时候,需要使用迭代器来遍历容器里面的元素;不同容器的迭代器有不同的类
型,在定义迭代器时必须指明。而迭代器的类型有时候比较复杂,请看下面的例子:
include
using namespace std;
int main(){
vector< vector > v;
//vector< vector >::iterator i = v.begin();
auto i = v.begin(); //使用 auto 代替具体的类型,该句比上一句简洁,根据表达式
v.begin() 的类型(begin() 函数的返回值类型)来推导出变量i的类型
return 0;
}
用于泛型编程
auto 的另一个应用就是当我们不知道变量是什么类型,或者不希望指明具体类型的时候,比如
泛型编程中。请看下面例子:
#include
using namespace std;
class A{
public:
static int get(void){
return 100;
}
};
class B{
public:
static const char* get(void){
return “www.123”;
}
};
template
void func(void){
auto val = T::get();
cout << val << endl;
}
int main(void){
func();
func();
return 0;
}
/*
运行结果:
100
www.123
*/
本例中的模板函数 func() 会调用所有类的静态函数 get(),并对它的返回值做统一处理,但是 get() 的
返回值类型并不一样,而且不能自动转换。这种要求在以前的 C++ 版本中实现起来非常的麻烦,需要额
外增加一个模板参数,并在调用时手动给该模板参数赋值,用以指明变量 val 的类型。 但是有了 auto
类型自动推导,编译器就根据 get() 的返回值自己推导出 val 变量的类型,就不用再增加一个模板参数
了。
1.5.13 简述一下 C++11 中的可变参数模板新特性
参考回答
可变参数模板(variadic template)使得编程者能够创建这样的模板函数和模板类,即可接受可变数量
的参数。例如要编写一个函数,它可接受任意数量的参数,参数的类型只需是cout能显示的即可,并将
参数显示为用逗号分隔的列表。
int n = 14;
double x = 2.71828;
std::string mr = “Mr.String objects!”;
show_list(n, x);
show_list(xx, ‘!’, 7, mr); //这里的目标是定义show_list()
/
运行结果:
14, 2.71828
7.38905, !, 7, Mr.String objects!
要创建可变参数模板,需要理解几个要点:
(1)模板参数包(parameter pack);
(2)函数参数包;
(3)展开(unpack)参数包;
(4)递归。
1.5.14 简述一下 C++11 中 Lambda 新特性
参考回答
定义
lambda 匿名函数很简单,可以套用如下的语法格式:
[外部变量访问方式说明符] (参数) mutable noexcept/throw() -> 返回值类型
{
函数体;
};
其中各部分的含义分别为:
a. [外部变量方位方式说明符]
[ ] 方括号用于向编译器表明当前是一个 lambda 表达式,其不能被省略。在方括号内部,可以
注明当前 lambda 函数的函数体中可以使用哪些“外部变量”。
所谓外部变量,指的是和当前 lambda 表达式位于同一作用域内的所有局部变量。
b. (参数)
和普通函数的定义一样,lambda 匿名函数也可以接收外部传递的多个参数。和普通函数不同的
是,如果不需要传递参数,可以连同 () 小括号一起省略;
c. mutable
此关键字可以省略,如果使用则之前的 () 小括号将不能省略(参数个数可以为 0)。默认情况
下,对于以值传递方式引入的外部变量,不允许在 lambda 表达式内部修改它们的值(可以理解
为这部分变量都是 const 常量)。而如果想修改它们,就必须使用 mutable 关键字。
注意:对于以值传递方式引入的外部变量,lambda 表达式修改的是拷贝的那一份,并不会修改
真正的外部变量;
d. noexcept/throw()
可以省略,如果使用,在之前的 () 小括号将不能省略(参数个数可以为 0)。默认情况下,
lambda 函数的函数体中可以抛出任何类型的异常。而标注 noexcept 关键字,则表示函数体内不
会抛出任何异常;使用 throw() 可以指定 lambda 函数内部可以抛出的异常类型。e. -> 返回值类型
指明 lambda 匿名函数的返回值类型。值得一提的是,如果 lambda 函数体内只有一个 return
语句,或者该函数返回 void,则编译器可以自行推断出返回值类型,此情况下可以直接省略"-> 返
回值类型"。
f. 函数体
和普通函数一样,lambda 匿名函数包含的内部代码都放置在函数体中。该函数体内除了可以使
用指定传递进来的参数之外,还可以使用指定的外部变量以及全局范围内的所有全局变量。
#include
#include
using namespace std;
int main()
{
int num[4] = {4, 2, 3, 1};
//对 a 数组中的元素进行排序
sort(num, num+4, [=](int x, int y) -> bool{ return x < y; } );
for(int n : num){
cout << n << " ";
}
return 0;
}
/*
程序运行结果:
1 2 3 4
*/
2. 操作系统
2.1 Linux中查看进程运行状态的指令、查看内存使用情况的指令、
tar解压文件的参数。
参考回答
查看进程运行状态的指令: ps命令。“ps -aux | grep PID”,用来查看某PID进程状态
查看内存使用情况的指令: free命令。“free -m”,命令查看内存使用情况。
tar解压文件的参数:
五个命令中必选一个
-c: 建立压缩档案
-x: 解压
-t: 查看内容
-r: 向压缩归档文件末尾追加文件
-u: 更新原压缩包中的文件
这几个参数是可选的
-z: 有gzip属性的
-j: 有bz2属性的
-Z: 有compress属性的
-v: 显示所有过程
-O: 将文件解开到标准输出
答案解析
//ps使用示例
//显示当前所有进程
ps -A
//与grep联用查找某进程
ps -aux | grep apache
//查看进程运行状态、查看内存使用情况的指令均可使用top指令。
top
2.2 文件权限怎么修改
参考回答
Linux文件的基本权限就有九个,分别是owner/group/others三种身份各有自己的read/write/execute
权限
修改权限指令: chmod
答案解析
举例:文件的权限字符为 -rwxrwxrwx 时,这九个权限是三个三个一组。其中,我们可以使用数字来代
表各个权限。
各权限的分数对照如下:
image-20220712215706188
每种身份(owner/group/others)各自的三个权限(r/w/x)分数是需要累加的,
例如当权限为: [-rwxrwx—] ,则分数是:
owner = rwx = 4+2+1 = 7
group = rwx = 4+2+1 = 7
others= — = 0+0+0 = 0
所以我们设定权限的变更时,该文件的权限数字就是770!变更权限的指令chmod的语法是这样的:
[root@www ~]# chmod [-R] xyz 文件或目录
选项与参数:
xyz : 就是刚刚提到的数字类型的权限属性,为 rwx 属性数值的相加。
-R : 进行递归(recursive)的持续变更,亦即连同次目录下的所有文件都会变更
chmod 770 test.c //即修改test.c文件的权限为770
2.3 说说常用的Linux命令
参考回答
cd命令:用于切换当前目录
ls命令:查看当前文件与目录
grep命令:该命令 常用于分析一行的信息, 若当中有我们所需要的信息,就将该行显示出来,该
命令通常与管道命令一起使用,用于对一些命令的输出进行筛选加工。
cp命令:复制命令
mv命令:移动文件或文件夹命令
rm命令:删除文件或文件夹命令
ps命令:查看进程情况
kill命令:向进程发送终止信号
tar命令:对文件进行打包,调用gzip或bzip对文件进行压缩或解压
cat命令:查看文件内容,与less、more功能相似
top命令:可以查看操作系统的信息,如进程、CPU占用率、内存信息等
pwd命令:命令用于显示工作目录。
2.4 说说如何以root权限运行某个程序。
参考回答
sudo chown root app(文件名)
sudo chmod u+s app(文件名)
输入上面两条指令后即可
2.5 说说软链接和硬链接的区别。
参考回答
定义不同
软链接 又叫 符号链接,这个文件 包含了 另一个文件的 路径名。可以是任意文件或目录,可以链接不
同文件系统的文件。
硬链接就是一个文件的 一个或多个文件名。把 文件名和计算机文件系统 使用的节点号链接起来。因
此我们可以用多个文件名与同一个文件进行链接,这些文件名 可以在同一目录或不同目录。
2.限制不同
硬链接 只能对 已存在的文件进行创建,不能交叉文件系统进行硬链接的创建;
软链接 可对 不存在的文件或目录创建 软链接; 可交叉文件系统;
3.创建方式不同
硬链接 不能 对目录进行创建,只可对文件创建;
软链接 可对 文件或目录创建;
4.影响不同
删除一个硬链接文件并不影响其他 有相同 inode 号的文件。
删除 软链接 并不影响被指向的文件,但若被指向的原文件被删除,则相关软连接被称为死链接(即
dangling link,若被指向路径文件被重新创建,死链接可恢复为正常的软链接)。
2.6 说说静态库和动态库怎么制作及如何使用,区别是什么。
参考回答
静态库的制作:
gcc
hello.c -c //这样就生成hello.o目标文件
ar rcs libhello.a hello.o//生成libhello.a静态库
静态库的使用:
gcc main.c -lhello -o staticLibrary//main.c和hello静态库链接,生成staticLibrary执行
文件
/*
main.c:是指main主函数
-lhello:是我们生成的.a 文件砍头去尾(lib不要 .a也不要)前面加-l
-L:是指告诉gcc编译器先从-L指定的路径去找静态库,默认是从/usr/lib/ 或者
/usr/local/lib/ 去找。
./:是指当前路径的意思
staticLibrary:是最后想生成的文件名(这里可随意起名字)
*/
动态库的制作:
gcc -shared -fpic hello.c -o libhello.so
-shared 指定生成动态库
-fpic :fPIC选项作用于编译阶段,在生成目标文件时就得使用该选项,以生成位置无关的代码。
动态库的使用:
gcc main.c -lhello -L ./ -o dynamicDepot
/*
main.c:是指main主函数
-lhello:是我们生成的.so 文件砍头去尾(lib不要 .so也不要)前面加-l
-L:是指告诉gcc编译器先从-L指定的路径去找静态库,默认是从/usr/lib/ 或者
/usr/local/lib/ 去找。
./:是指当前路径的意思
dynamicDepot:是最后想生成的文件名(这里可随意起名字)
*/
区别:
静态库代码装载的速度快,执行速度略比动态库快。
动态库更加节省内存,可执行文件体积比静态库小很多。
静态库是在编译时加载,动态库是在运行时加载。
生成的静态链接库,Windows下以.lib为后缀,Linux下以.a为后缀。生成的动态链接库,
Windows下以.dll为后缀,Linux下以.so为后缀。
2.7 简述GDB常见的调试命令,什么是条件断点,多进程下如何调
试。
参考回答
GDB调试: gdb调试的是可执行文件,在gcc编译时加入 -g ,告诉gcc在编译时加入调试信息,这样gdb
才能调试这个被编译的文件 gcc -g tesst.c -o test
GDB命令格式:
quit:退出gdb,结束调试
list:查看程序源代码
list 5,10:显示5到10行的代码
list test.c:5, 10: 显示源文件5到10行的代码,在调试多个文件时使用
list get_sum: 显示get_sum函数周围的代码
list test,c get_sum: 显示源文件get_sum函数周围的代码,在调试多个文件时使用
reverse-search:字符串用来从当前行向前查找第一个匹配的字符串
run:程序开始执行
help list/all:查看帮助信息
break:设置断点
break 7:在第七行设置断点
break get_sum:以函数名设置断点
break 行号或者函数名 if 条件:以条件表达式设置断点
watch 条件表达式:条件表达式发生改变时程序就会停下来
next:继续执行下一条语句 ,会把函数当作一条语句执行
step:继续执行下一条语句,会跟踪进入函数,一次一条的执行函数内的代码
条件断点: break if 条件 以条件表达式设置断点
多进程下如何调试:用set follow-fork-mode child 调试子进程
或者set follow-fork-mode parent 调试父进程
2.8 说说什么是大端小端,如何判断大端小端?
小端模式:低的有效字节存储在低的存储器地址。小端一般为主机字节序;常用的X86结构是小端模
式。很多的ARM,DSP都为小端模式。
大端模式:高的有效字节存储在低的存储器地址。大端为网络字节序;KEIL C51则为大端模式。
有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。
如何判断:我们可以根据联合体来判断系统是大端还是小端。因为联合体变量总是从低地址存储。
int fun1(){
union test{
char c;
int i;
};
test t; t.i = 1;
//如果是大端,则t.c为0x00,则t.c != 1,反之是小端
return (t.c == 1);
}
答案解析
在进行网络通信时是否需要进行字节序转换?
相同字节序的平台在进行网络通信时可以不进行字节序转换,但是跨平台进行网络数据通信时必须
进行字节序转换。
原因如下:网络协议规定接收到得第一个字节是高字节,存放到低地址,所以发送时会首先去低地
址取数据的高字节。小端模式的多字节数据在存放时,低地址存放的是低字节,而被发送方网络协
议函数发送时会首先去低地址取数据(想要取高字节,真正取得是低字节),接收方网络协议函数
接收时会将接收到的第一个字节存放到低地址(想要接收高字节,真正接收的是低字节),所以最
后双方都正确的收发了数据。而相同平台进行通信时,如果双方都进行转换最后虽然能够正确收发
数据,但是所做的转换是没有意义的,造成资源的浪费。而不同平台进行通信时必须进行转换,不
转换会造成错误的收发数据,字节序转换函数会根据当前平台的存储模式做出相应正确的转换,如
果当前平台是大端,则直接返回不进行转换,如果当前平台是小端,会将接收到得网络字节序进行
转换。
网络字节序
网络上传输的数据都是字节流,对于一个多字节数值,在进行网络传输的时候,先传递哪个字节?也就
是说,当接收端收到第一个字节的时候,它将这个字节作为高位字节还是低位字节处理,是一个比较有
意义的问题; UDP/TCP/IP协议规定:把接收到的第一个字节当作高位字节看待,这就要求发送端发送
的第一个字节是高位字节;而在发送端发送数据时,发送的第一个字节是该数值在内存中的起始地址
处对应的那个字节,也就是说,该数值在内存中的起始地址处对应的那个字节就是要发送的第一个高
位字节(即:高位字节存放在低地址处);由此可见,多字节数值在发送之前,在内存中因该是以大端法存
放的; 所以说,网络字节序是大端字节序; 比如,我们经过网络发送整型数值0x12345678时,在80X86
平台中,它是以小端发存放的,在发送之前需要使用系统提供的字节序转换函数htonl()将其转换成大
端法存放的数值;
2.9 说说进程调度算法有哪些?
参考回答
先来先服务调度算法
短作业(进程)优先调度算法
高优先级优先调度算法
时间片轮转法
多级反馈队列调度算法
答案解析
先来先服务调度算法:每次调度都是从后备作业(进程)队列中选择一个或多个最先进入该队列的
作业(进程),将它们调入内存,为它们分配资源、创建进程,然后放入就绪队列。
短作业(进程)优先调度算法:短作业优先(SJF)的调度算法是从后备队列中选择一个或若干个估计运
行时间最短的作业(进程),将它们调入内存运行。
高优先级优先调度算法:当把该算法用于作业调度时,系统将从后备队列中选择若干个优先权最高
的作业装入内存。当用于进程调度时,该算法是把处理机分配给就绪队列中优先权最高的进程
时间片轮转法:每次调度时,把CPU 分配给队首进程,并令其执行一个时间片。时间片的大小从
几ms 到几百ms。当执行的时间片用完时,由一个计时器发出时钟中断请求,调度程序便据此信
号来停止该进程的执行,并将它送往就绪队列的末尾;然后,再把处理机分配给就绪队列中新的队
首进程,同时也让它执行一个时间片。
多级反馈队列调度算法:综合前面多种调度算法。
在这些调度算法中,有抢占式和非抢占式的区别
非抢占式优先权算法
在这种方式下,系统一旦把处理机分配给就绪队列中优先权最高的进程后,该进程便一直执行下
去,直至完成;或因发生某事件使该进程放弃处理机时,系统方可再将处理机重新分配给另一优先
权最高的进程。这种调度算法主要用于批处理系统中;也可用于某些对实时性要求不严的实时系统
中。
抢占式优先权调度算法
在这种方式下,系统同样是把处理机分配给优先权最高的进程,使之执行。但在其执行期间,只要
又出现了另一个其优先权更高的进程,进程调度程序就立即停止当前进程(原优先权最高的进程)的
执行,重新将处理机分配给新到的优先权最高的进程。因此,在采用这种调度算法时,是每当系统
中出现一个新的就绪进程i 时,就将其优先权Pi与正在执行的进程j 的优先权Pj进行比较。如果
Pi≤Pj,原进程Pj便继续执行;但如果是Pi>Pj,则立即停止Pj的执行,做进程切换,使i 进程投入执
行。显然,这种抢占式的优先权调度算法能更好地满足紧迫作业的要求,故而常用于要求比较严格
的实时系统中,以及对性能要求较高的批处理和分时系统中。
区别:
非抢占式(Nonpreemptive):让进程运行直到结束或阻塞的调度方式,容易实现,适合专用系统,不
适合通用系统。
抢占式(Preemptive):允许将逻辑上可继续运行的在运行过程暂停的调度方式可防止单一进程长时
间独占,CPU系统开销大(降低途径:硬件实现进程切换,或扩充主存以贮存大部分程序)
2.10 简述操作系统如何申请以及管理内存的?
参考回答
物理内存:物理内存有四个层次,分别是寄存器、高速缓存、主存、磁盘。
寄存器:速度最快、量少、价格贵。
高速缓存:次之。
主存:再次之。
磁盘:速度最慢、量多、价格便宜。
image-20220713132002140
操作系统会对物理内存进行管理,有一个部分称为内存管理器(memory manager),它的主要工
作是有效的管理内存, 记录哪些内存是正在使用的, 在进程需要时分配内存 以及在进程完成时回收
内存。
虚拟内存:
操作系统为 每一个进程 分配 一个独立的地址空间,但是虚拟内存。虚拟内存与物理内存
存在映射关系,通过 页表寻址 完成虚拟地址和物理地址的转换。
操作系统如何申请内存:
从操作系统角度来看,进程分配内存有两种方式,分别由两个 系统调用 完成:*brk 和 mmap
2.11 简述Linux系统态与用户态,什么时候会进入系统态?
参考回答
内核态与用户态: 内核态(系统态) 与 用户态 是操作系统的两种运行级别。内核态拥有最高权限,
可以访问所有系统指令; 用户态则只能访问一部分指令。
什么时候进入内核态:共有三种方式:
a、系统调用。
b、异常。
c、设备中断。
其中,系统调用是主动的 , 另外 两种是被动的。
为什么区分内核态与用户态:
在CPU的所有指令中,有一些指令是非常危险的,如果错用,将导致
整个系统崩溃。比如:清内存、设置时钟等。所以区分内核态与用户态主要是出于安全的考虑。
2.12 简述LRU算法及其实现方式。
参考回答
LRU算法:LRU算法用于缓存淘汰。思路 是将缓存中最近最少使用的对象删除掉
实现方式:利用链表和hashmap。
当需要 插入新的数据项的时候,如果新数据项在链表中存在(一般称为命中),则把该节点移到链
表头部,如果不存在,则新建一个节点,放到链表头部,若缓存满了,则把链表最后一个节点删除
即可。
在访问数据的时候,如果数据项在链表中存在,则把该节点移到链表头部, 否则返回-1。这样一来
在链表尾部的节点就是最近最久未访问的数据项。
答案解析
给出C++实现的代码
class LRUCache {
list<pair<int, int>> cache;//创建双向链表
unordered_map<int, list<pair<int, int>>::iterator> map;//创建哈希表
int cap;
public:
LRUCache(int capacity) {
cap = capacity;
}
int get(int key) {
if (map.count(key) > 0){
auto temp = *map[key];
cache.erase(map[key]);
map.erase(key);
cache.push_front(temp); map[key] = cache.begin();//映射头部
return temp.second;
}
return -1;
}
void put(int key, int value) {
if (map.count(key) > 0){
cache.erase(map[key]);
map.erase(key);
}
else if (cap == cache.size()){
auto temp = cache.back();
map.erase(temp.first);
cache.pop_back();()
}
cache.push_front(pair<int, int>(key, value));
map[key] = cache.begin();//映射头部
}
};
/**
- Your LRUCache object will be instantiated and called as such:
- LRUCache* obj = new LRUCache(capacity);
- int param_1 = obj->get(key);
- obj->put(key,value);
*/
2.13 一个线程占多大内存?
参考回答
一个linux的线程大概占8M内存。
答案解析
linux的栈 是通过缺页来分配内存的, 不是**所有 栈地址空间都分配了内存。因此,8M是最大消耗,实际
的内存消耗只会略大于实际需要的内存(内部损耗,每个在4k以内)***。
2.14 什么是页表,为什么要有?
参考回答
页表是虚拟内存的概念。操作系统 虚拟内存 到 物理内存 的映射表,就被称为页表。
原因:不可能每一个虚拟内存的 Byte 都对应到物理内存的地址。这张表将大得真正的物理地址也放不
下,于是操作系统引入了页(Page)的概念。进行分页,这样可以减小虚拟内存页对应物理内存页的映
射表大小。
答案解析
如果将每一个虚拟内存的 Byte 都对应到物理内存的地址,每个条目最少需要 8字节(32位虚拟地址-
32位物理地址),在 4G 内存的情况下,就需要 32GB 的空间来存放对照表,那么这张表就大得真正
的物理地址也放不下了,于是操作系统引入了页(Page)的概念。
在系统启动时,操作系统 将 整个物理内 存以 4K 为单位,划分为各个页。之后 进行内存分配时,都以页
为单位,那么 虚拟内存页对应物理内存页的映射表就大大减小了, 4G 内存,只需要 8M 的映射表即
可,一些进程没有使用到的虚拟内存,也并不需要保存映射关系,而且Linux 还为大内存设计了多级页
表,可以进一页减少了内存消耗。
2.15 简述操作系统中的缺页中断。
参考回答
缺页异常: malloc和mmap函数在分配内存时只是建立了进程虚拟地址空间,并没有 分配虚拟内
存对应的物理内存。当 进程访问这些 没有建立映射关系的虚拟内存时,处理器自动触发一个缺页异
常,引发缺页中断。
缺页中断:缺页异常后 将产生一个缺页中断,此时 操作系统会根据页表中的外存地址在外存中找到
所缺的一页,将其调入内存。
答案解析
两者区别。
缺页中断 与一般中断一样 ,需要经历四个步骤:保护CPU现场、分析中断原因、转入缺页中断处理程
序、恢复CPU现场,继续执行。
缺页中断 与 一般中断区别:
(1)在指令执行期间 产生和处理 缺页中断信号
(2)一条指令 在执行期间,可能产生多次缺页中断
(3)缺页中断 返回的是 执行产生中断的一条指令,而一般中断返回 的是执行下一条指令。
2.16 说说虚拟内存分布,什么时候会由用户态陷入内核态?
参考回答
虚拟内存分布:
image-20220713140624628
用户空间:
(1)代码段.text:存放程序执行代码的一块内存区域。只读,代码段的头部还会包含一些只读的
常数变量。
(2)数据段.data:存放程序中已初始化的全局变量和静态变量的一块内存区域。
(3)BSS 段.bss:存放程序中未初始化的全局变量和静态变量的一块内存区域。
(4)可执行程序在运行时又会多出两个区域:堆区和栈区。
堆区:动态申请内存用。堆从低地址向高地址增长。
栈区:存储局部变量、函数参数值。栈从高地址向低地址增长。是一块连续的空间。
(5)最后还有一个共享区,位于堆和栈之间。
内核空间:DMA区、常规区、高位区。
什么时候进入内核态:共有三种方式:
a、系统调用。b、异常。c、设备中断。其中,系统调用是主动的,另外两种是被动的。
2.17 简述一下虚拟内存和物理内存,为什么要用虚拟内存,好处是
参考回答
物理内存:物理内存有四个层次,分别是寄存器、高速缓存、主存、磁盘。
寄存器:速度最快、量少、价格贵。高速缓存:次之。
主存:再次之。
磁盘:速度最慢、量多、价格便宜。
操作系统会对物理内存进行管理,有一个部分称为内存管理器(memory manager),它的主要工
作是有效的管理内存,记录哪些内存是正在使用的,在进程需要时分配内存以及 在进程完成时回收
内存。
虚拟内存:操作系统 为每一个进程分配一个独立的地址空间,但是虚拟内存。虚拟内存与物理内存
存在映射关系,通过页表寻址完成虚拟地址和物理地址的转换。
为什么要用虚拟内存:因为早期的内存分配方法存在以下问题:
(1)进程地址空间不隔离。会导致数据被随意修改。
(2)内存使用效率低。
(3)程序运行的地址不确定。操作系统随机为进程分配内存空间,所以程序运行的地址是不确定
的。
使用虚拟内存的好处:
(1)扩大地址空间。每个进程独占一个4G空间,虽然真实物理内存没那么多。
(2)内存保护:防止不同进程对物理内存的争夺和践踏,可以对特定内存地址提供写保护,防止
恶意篡改。
(3)可以实现内存共享,方便进程通信。
(4)可以避免内存碎片,虽然物理内存可能不连续,但映射到虚拟内存上可以连续。
使用虚拟内存的缺点:
(1)虚拟内存 需要额外构建数据结构,占用空间。
(2)虚拟地址到物理地址的转换,增加了执行时间。
(3)页面换入换出 耗时。
(4)一页如果只有一部分数据,浪费内存。
2.18 虚拟地址到物理地址怎么映射的?
参考回答
操作系统为每一个进程维护了一个从虚拟地址到物理地址的映射关系的数据结构,叫页表。页表中的每
一项都记录了这个页的基地址。
三级页表转换方法:(两步)
逻辑地址转线性地址: 段起始地址+段内偏移地址=线性地址
线性地址 转 物理地址:
(1)每一个32位的线性地址被划分为三部分:页目录索引(DIRECTORY,10位)、页表索引
(TABLE,10位)、页内偏移(OFFSET,12位)
(2)从cr3中取出进程的页目录地址(操作系统调用进程时,这个地址被装入寄存器中)
页目录地址 + 页目录索引 = 页表地址
页表地址 + 页表索引 = 页地址
页地址 + 页内偏移 = 物理地址
image-20220713141503280
按照以上两步法,就完成了一个三级页表从虚拟地址到物理地址的转换。
2.19 说说堆栈溢出是什么,会怎么样?
参考回答
堆栈溢出 就是 不顾堆栈中分配的局部数据块大小,向该数据块写入了过多的数据,导致数据越界。常指
调用堆栈溢出,本质上一种数据结构的满溢情况。堆栈溢出可以理解为两个方面:堆溢出和栈溢出。
堆溢出:比如不断的new 一个对象,一直创建新的对象,而不进行释放,最终导致内存不足。将
会报错:OutOfMemory Error。
栈溢出: 一次函数调用中,栈中将被依次压入:参数,返回地址等,而方法 如果递归比较深或进去
死循环,就会导致栈溢出。将会报错:StackOverflow Error。
2.20 简述操作系统中malloc的实现原理
参考回答
malloc底层实现: 当开辟的空间小于 128K 时,调用 brk()函数;当开辟的空间大于 128K 时,调用
mmap()。malloc采用的是内存池的管理方式,以减少内存碎片。先申请大块内存作为堆区,然后将
堆区分为多个内存块。当用户申请内存时,直接从堆区分配一块合适的空闲快。采用隐式链表将所有空
闲块,每一个空闲块 记录了一个未分配的、连续的内存地址。
2.21 说说进程空间从高位到低位都有些什么?
参考回答
image-20220713142634083
如上图,从高地址到低地址,一个程序由 命令行参数和环境变量、栈、文件映射区、堆、BSS段、数据
段、代码段组成。
命令行参数和环境变量
栈区:存储局部变量、函数参数值。栈从高地址向低地址增长。是一块连续的空间。
文件映射区,位于堆和栈之间。
堆区:动态申请内存用。堆从低地址向高地址增长。
BSS 段:存放程序中未初始化的全局变量和静态变量的一块内存区域。
数据段:存放程序中已初始化的全局变量和静态变量的一块内存区域。
代码段:存放程序执行代码的一块内存区域。只读,代码段的头部还会包含一些只读的常数变量。
2.22 32位系统能访问4GB以上的内存吗?
参考回答
正常情况下是不可以的。原因是计算机使用二进制,每位数只有0或1两个状态,32位正好是2的32次
方,正好是4G, 所以大于4G就没办法表示了,而在32位的系统中,因其它原因还需要占用一部分空
间,所以内存只能识别3G多。要使用4G以上就只能换64位的操作系统了。
但是使用PAE技术就可以实现 32位系统能访问4GB以上的内存。
答案解析
**
Physical Address Extension(PAE)技术 最初是为了弥补32位地址在PC服务器应用上的不足而推出
的。 我们知道,传统的IA32架构只有32位地址总线,只能让系统容纳不超过4GB的内存,这么大的内
存,对于普通的桌面应用应该说是足够用了。可是,对于服务器应用来说,还是显得不足,因为服务器
上可能承载了很多同时运行的应用。PAE技术将地址扩展到了36位,这样,系统就能够容纳2^36=64GB
的内存。
2.23 请你说说并发和并行
参考回答
并发: 对于单个CPU,在一个时刻只有一个进程在运行,但是线程的切换时间则减少到纳秒数量
级,多个任务不停来回快速切换。
并行:对于 多个CPU,多个进程同时运行。
区别。通俗来讲,它们虽然都说是"多个进程同时运行",但是它们的"同时"不是一个概念。并行
的"同时"是同一时刻可以多个任务在运行(处于running), 并发的"同时"是经过不同线程快速切换,
使得看上去多个任务同时都在运行的现象。
2.24 说说进程、线程、协程是什么,区别是什么?
参考回答
进程: 程序是指令、数据及其组织形式的描述,而进程则是程序的运行实例,包括程序计数器、寄
存器和变量的当前值。
线程: 微进程,一个进程里更小粒度的执行单元。一个进程里包含多个线程并发执行任务。
协程:协程是微线程,在子程序内部执行,可在子程序内部中断,转而执行别的子程序,在适当的
时候再返回来接着执行。
区别:
线程与线程的区别:
(1)一个线程从属于一个进程;一个进程可以包含多个线程。
(2)一个线程挂掉,对应的进程挂掉;一个进程挂掉,不会影响其他进程。
(3)进程是系统资源调度的最小单位;线程CPU调度的最小单位。
(4)进程系统开销 显著大于 线程开销; 线程需要的系统资源更少。
(5)进程 在执行时拥有独立的内存单元,多个线程 共享进程的内存 ,如代码段、数据段、扩展
段;但 每个线程 拥有自己的栈段和寄存器组。
(6)进程切换时需要刷新TLB并获取新的地址空间,然后切换硬件上下文和内核栈, 线程切换时
只需要切换硬件上下文和内核栈。
(7)通信方式不一样。
(8)进程适应于多核、多机分布; 线程适用于多核
线程与协程的区别:
(1)协程执行效率极高。协程直接操作栈基本没有内核切换的开销,所以上下文的切换非常快,
切换开销比线程更小。
(2)协程 不需要多线程的锁机制,因为多个协程从属于一个线程,不存在同时写变量冲突,效率
比线程高。
(3)一个线程可以有多个协程。
2.25 请你说说Linux的fork的作用
参考回答
fork函数用来创建一个子进程。对于父进程,fork()函数返回新创建的子进程的PID。对于子进程,
fork()函数调用成功会返回0。如果创建出错,fork()函数返回-1。
答案解析
fork()函数,其原型如下:
#include <unistd.h>
pid_t fork(void);
fork()函数不需要参数,返回值是一个进程标识符PID。返回值有以下三种情况:
(1) 对于父进程,fork()函数返回新创建的子进程的PID。
(2) 对于子进程,fork()函数调用成功会返回0。
(3) 如果创建出错,fork()函数返回-1。
fork()函数创建一个新进程后,会为这个新进程分配进程空间,将 父进程的进程空间中的内容 复制到 子
进程的进程空间中,包括父进程的数据段和堆栈段,并且和父进程共享代码段。这时候,子进程和父进
程一模一样,都接受系统的调度。因为两个进程都停留在fork()函数中,最后fork()函数会返回两次,一
次在父进程中返回,一次在子进程中返回,两次返回的值不一样,如上面的三种情况。
2.26 请你说说什么是孤儿进程,什么是僵尸进程,如何解决僵尸进程
参考回答
孤儿进程:是指一个父进程退出后,而它的一个或多个子进程还在运行,那么这些子进程将成为孤
儿进程。孤儿进程将被init进程(进程号为1)所收养,并且由init进程对它们完整状态收集工作。
僵尸进程:是指一个进程使用fork函数创建子进程,如果子进程退出,而父进程并没有调用wait()
或者waitpid()系统调用取得子进程的终止状态,那么子进程的进程描述符仍然保存在系统中,占
用系统资源,这种进程称为僵尸进程。
如何解决僵尸进程:
(1)一般,为了防止产生僵尸进程,在fork子进程之后我们都要及时使用wait系统调用;同时,
当子进程退出的时候,内核都会给父进程一个SIGCHLD信号,所以我们可以建立一个捕获
SIGCHLD信号的信号处理函数,在函数体中调用wait(或waitpid),就可以清理退出的子进程以
达到防止僵尸进程的目的。
(2)使用kill命令。
打开终端并输入下面命令:
ps aux | grep Z
会列出进程表中所有僵尸进程的详细内容。
然后输入命令:
kill -s SIGCHLD pid(父进程pid)
2.27 请你说说什么是守护进程,如何实现?
参考回答
守护进程: 守护进程是运行在后台的一种生存期长的特殊进程。它独立于控制终端,处理一些系统
级别任务。
如何实现:
(1)创建 子进程,终止 父进程。方法是调用fork() 产生一个子进程,然后使父进程退出。
(2)调用 setsid() 创建一个新会话。
(3)将 当前目录 更改为 根目录。使用fork() 创建的子进程也继承了 父进程的当前工作目录。
(4)重设 文件权限 掩码。文件权限掩码 是指屏蔽掉文件权限中的对应位。
(5)关闭 不再需要的文件描述符。子进程 从 父进程 继承打开的文件描述符。
答案解析
实现代码如下:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <sys/stat.h>
#define MAXFILE 65535
int main(){
//第一步:创建进程
int pid = fork();
if (pid > 0)
exit(0);//结束父进程
else if (pid < 0){
printf(“fork error!\n”);
exit(1);//fork失败,退出
}
//第二步:子进程成为 新的会话组长和进程组长,并与控制终端分离
setsid();
//第三步:改变工作目录到
chdir(“/”);
//第四步:重设文件创建掩模
umask(0);
//第五步:关闭打开的文件描述符
for (int i=0; i<MAXFILE; ++i)
close(i);
sleep(2);
}
return 0;
}
2.28 说说进程通信的方式有哪些?
参考回答
进程间通信主要包括 管道、系统IPC(包括消息队列、信号量、信号、共享内存)、套接字socket
管道: 包括无名管道和命名管道, 无名管道半双工,只能用于具有亲缘关系的进程直接的通信(父
子进程或者兄弟进程),可以看作一种特殊的文件; 命名管道可以允许无亲缘关系进程间的通信。
系统IPC
消息队列:消息的链接表,放在内核中。消息队列独立于发送与接收进程,进程终止时,消息队列
及其内容并不会被删除;消息队列可以实现消息的随机查询,可以按照消息的类型读取。
信号量semaphore:是一个计数器,可以用来控制多个进程对共享资源的访问。信号量用于实现
进程间的互斥与同步。
信号:用于通知接收进程某个事件的发生。
内存共享:使多个进程访问同一块内存空间。
套接字socket:用于不同主机直接的通信。
2.29 说说进程同步的方式?
参考回答
信号量semaphore: 是一个计数器,可以用来控制多个进程对共享资源的访问。信号量用于实现
进程间的互斥与同步。P操作(递减操作)可以用于阻塞一个进程, V操作(增加操作)可以用于解除阻
塞一个进程。
管道: 一个进程通过 调用管程的一个过程 进入管程。在任何时候,只能有一个进程在管程中执行,
调用管程的任何 其他进程都被阻塞,以等待管程可用。
消息队列:消息的链接表,放在内核中。消息队列 独立于发送与接收进程,进程终止时,消息队列
及其内容并不会被删除;
2.30 说说Linux进程调度算法及策略有哪些?
参考回答
先来先服务调度算法
短作业(进程)优先调度算法
高优先级优先调度算法
时间片轮转法
多级反馈队列调度算法
答案解析
先来先服务调度算法:每次调度都是从后备作业(进程)队列中选择一个或多个最先进入该队列的
作业(进程),将它们调入内存,为它们分配资源、创建进程,然后放入就绪队列。
短作业(进程)优先调度算法:短作业优先(SJF)的调度算法是从后备队列中选择一个或若干个估计运
行时间最短的作业(进程),将它们调入内存运行。
高优先级优先调度算法:当把该算法用于作业调度时,系统将从后备队列中选择若干个优先权最高
的作业装入内存。当用于进程调度时,该算法是把处理机分配给就绪队列中优先权最高的进程
时间片轮转法:每次调度时,把CPU 分配给队首进程,并令其执行一个时间片。时间片的大小从
几ms 到几百ms。当执行的时间片用完时,由一个计时器发出时钟中断请求,调度程序便据此信
号来停止该进程的执行,并将它送往就绪队列的末尾;然后,再把处理机分配给就绪队列中新的队
首进程,同时也让它执行一个时间片。
多级反馈队列调度算法:综合前面多种调度算法。
在这些调度算法中,有抢占式和非抢占式的区别。
非抢占式优先权算法
在这种方式下,系统一旦把处理机分配给就绪队列中优先权最高的进程后,该进程便一直执行下
去,直至完成;或因发生某事件使该进程放弃处理机时,系统方可再将处理机重新分配给另一优先
权最高的进程。这种调度算法主要用于批处理系统中;也可用于某些对实时性要求不严的实时系统
中。
抢占式优先权调度算法
在这种方式下,系统同样是把处理机分配给优先权最高的进程,使之执行。但在其执行期间,只要
又出现了另一个其优先权更高的进程,进程调度程序就立即停止当前进程(原优先权最高的进程)的
执行,重新将处理机分配给新到的优先权最高的进程。因此,在采用这种调度算法时,是每当系统
中出现一个新的就绪进程i 时,就将其优先权Pi与正在执行的进程j 的优先权Pj进行比较。如果
Pi≤Pj,原进程Pj便继续执行;但如果是Pi>Pj,则立即停止Pj的执行,做进程切换,使i 进程投入执
行。显然,这种抢占式的优先权调度算法能更好地满足紧迫作业的要求,故而常用于要求比较严格
的实时系统中,以及对性能要求较高的批处理和分时系统中。
区别:
非抢占式(Nonpreemptive):让进程运行直到结束或阻塞的调度方式,容易实现,适合专用系统,不
适合通用系统。
抢占式(Preemptive):允许将逻辑上可继续运行的在运行过程暂停的调度方式可防止单一进程长时
间独占,CPU系统开销大(降低途径:硬件实现进程切换,或扩充主存以贮存大部分程序)
2.31 说说进程有多少种状态?
参考回答
进程有五种状态:创建、就绪、执行、阻塞、终止。一个进程创建后,被放入队列处于就绪状态,等待
操作系统调度执行,执行过程中可能切换到阻塞状态(并发),任务完成后,进程销毁终止。
答案解析
创建状态
一个应用程序从系统上启动,首先就是进入创建状态,需要获取系统资源创建进程管理块(PCB:
Process Control Block)完成资源分配。
就绪状态
在创建状态完成之后,进程已经准备好,处于就绪状态,但是还未获得处理器资源,无法运行。
运行状态
获取处理器资源,被系统调度,当具有时间片开始进入运行状态。如果进程的时间片用完了就进入就绪
状态。
阻塞状态
在运行状态期间,如果进行了阻塞的操作,如耗时的I/O操作,此时进程暂时无法操作就进入到了阻塞
状态,在这些操作完成后就进入就绪状态。等待再次获取处理器资源,被系统调度,当具有时间片就进
入运行状态。
终止状态
进程结束或者被系统终止,进入终止状态
image-20220713155449592
2.32 进程通信中的管道实现原理是什么?
参考回答
操作系统在 内核中开辟一块缓冲区(称为管道)用于通信。
管道是一种 两个进程间进行单向通信的机制。因为这种单向性,管道又称为半双工管道,所以其使用是有一定的局限性的。
半双工是指数据只能由一个进程流向另一个进程(一个管道负责读,一个管道负责写);如果是全双工通信,需要建立两个
管道。
管道分为无名管道和命名管道, 无名管道只能用于具有亲缘关系的进程直接的通信(父子进程或
者兄弟进程),可以看作一种特殊的文件,管道本质是一种文件;命名管道可以允许无亲缘关系进程间
的通信。
管道原型如下:
#include <unistd.h>
int pipe(int fd[2]);
pipe()函数 创建的管道 处于一个进程中间,因此 一个进程在由 pipe()创建管道后, 一般再使用fork() 建立
一个子进程,然后通过管道实现父子进程间的通信。管道两端 可分别用描述字fd[0]以及fd[1]来描述。
注意管道的两端的任务是固定的,即一端只能用于读,由描述字fd[0]表示,称其为管道读端; 另 一端
则只能用于写,由描述字fd[1]来表示,称其为管道写端。如果试图从 管道写端 读取数据,或者向 管道读
端写入数据都将发生错误。一般文件的 I/O 函数都可以用于管道,如close()、read()、write()等。
具体步骤如下:
父进程 调用pipe开辟管道,得到两个文件描述符指向 管道的两端。
父进程 调用fork创建子进程,那么子进程也有两个文件描述符指向同一管道。
父进程 关闭管道读端, 子进程关闭管道写端。 父进程可以往管道里写,子进程可以从管道里读,管道是
用环形队列实现的,数据从写端流入从读端流出,这样就实现了进程间通信。
答案解析
给出实现的代码,实现父子进程间的管道通信
#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#define INPUT
0
#define OUTPUT 1
int main(){ //创建管道
int fd[2];
pipe(fd);
//创建子进程
pid_t pid = fork();
if (pid < 0){
printf(“fork error!\n”);
exit(-1);
}
else if (pid == 0){//执行子进程
printf(“Child process is starting…\n”);
//子进程向父进程写数据,关闭管道的读端
close(fd[INPUT]);
write(fd[OUTPUT], “hello douya!”, strlen(“hello douya!”));
exit(0);
}
else{//执行父进程
printf (“Parent process is starting…\n”);
//父进程从管道读取子进程写的数据 ,关闭管道的写端
close(fd[OUTPUT]);
char buf[255];
int output = read(fd[INPUT], buf, sizeof(buf));
printf(“%d bytes of data from child process: %s\n”, output, buf);
}
return 0;
}
2.33 简述mmap的原理和使用场景
参考回答
原理: mmap是一种内存映射文件的方法, 即将一个文件或者 其它对象 映射到进程的地址空间,实现 文
件磁盘地址 和 进程虚拟地址空间中一段虚拟地址的一一对映关系。 实现这样的映射关系后, 进程就可以
采用指针的方式读写操作这一段内存,而 系统 会自动回写脏页面到对应的文件磁盘上,即完成了对文件
的操作 而不必再调用read, write等系统调用函数。 相反,内核空间对这段区域的修改 也直接 反映用户空
间,从而可以实现不同进程间的文件共享。如下图:
image-20220713160254112
使用场景:
对 同一块区域 频繁读写操作;
可用于实现 用户空间和内核空间的 高效交互
可提供进程间共享内存及相互通信
可实现高效的大规模数据传输。
2.34 互斥量能不能在进程中使用?
参考回答
能。
不同的进程之间 ,存在资源竞争或并发使用的问题,所以 需要互斥量。
进程中也需要互斥量,因为一个进程中可以包含多个线程, 线程与线程之间需要通过互斥的手段进行同
步,避免导致共享数据修改引起冲突。可以使用互斥锁,属于互斥量的一种。
2.35 协程是轻量级线程,轻量级表现在哪里?
参考回答
协程调用 跟 切换 比线程效率高: 协程 执行效率极高。协程不需要多线程的锁机制,可以不加锁的访
问全局变量,所以上下文的切换非常快。
协程占用内存少: 执行协程只需要极少的栈内存(大概是4~5KB),而默认情况下, 线程栈的大小为1MB。
切换开销更少:协程 直接操作 栈基本没有内核切换的开销,所以切换开销比线程少。
2.36 说说常见信号有哪些,表示什么含义?
参考回答
编号为1 ~ 31的信号 为传统UNIX支持的信号,是不可靠信号(非实时的)。
可靠信号和可靠信号的区别
在于前者不支持排队,可能会造成信号丢失,而后者不会。编号为1 ~ 31的信号如下:
image-20220713160754787
image-20220713160813842
image-20220713160843547
image-20220713160921956
2.37 说说线程间通信的方式有哪些?
参考回答
线程间的通信方式包括 临界区、互斥量、信号量、条件变量、读写锁:
临界区: 每个线程中访问临界资源的那段代码称为临界区(Critical Section)(临界资源是一次仅
允许一个线程使用的共享资源)。每次只准许一个线程进入临界区,进入后不允许其他线程进入。
不论是硬件临界资源,还是软件临界资源,多个线程 必须互斥地对它进行访问。
互斥量:采用互斥对象机制,只有拥有互斥对象的线程才可以访问。因为互斥对象只有一个,所以
可以保证公共资源不会被多个线程同时访问。
信号量:计数器,允许多个线程同时访问同一个资源。
条件变量:通过 条件变量 通知操作的方式来保持多线程同步。
读写锁: 读写锁 与 互斥量类似。但互斥量要么是锁住状态,要么就是不加锁状态。读写锁一次只允
许一个线程写,但允许一次多个线程读,这样效率就比互斥锁要高。
2.38 说说线程同步方式有哪些?
线程间的同步方式包括互斥锁、信号量、条件变量、读写锁:
互斥锁: 采用互斥对象机制,只有拥有互斥对象的线程才可以访问。因为互斥对象只有一个,所以
可以保证公共资源不会被多个线程同时访问。
信号量: 计数器,允许多个线程同时访问同一个资源。
条件变量: 通过 条件变量通知操作的方式 来保持 多线程同步。
读写锁: 读写锁与互斥量类似。但互斥量要么是锁住状态,要么就是不加锁状态。读写锁一次只允
许一个线程写,但允许一次多个线程读,这样效率就比互斥锁要高。
2.39 说说什么是死锁,产生的条件,如何解决?
参考回答
死锁: 是指 多个进程在执行过程中,因争夺资源而造成了互相等待。此时系统产生了死锁。比如两
只羊过独木桥,若两只羊互不相让,争着过桥,就产生死锁。
产生的条件:死锁发生有四个必要条件:
(1)互斥条件:进程 对所分配到的资源 不允许其他进程访问, 若其他进程访问,只能等待,直到
进程使用完成后释放该资源;
(2) 请求保持条件: 进程获得一定资源后,又对其他资源发出请求,但该资源被其他进程占有,
此时请求阻塞,而且该进程不会释放自己已经占有的资源;
(3)不可剥夺条件: 进程已获得的资源,只能自己释放,不可剥夺;
(4)环路等待条件: 若 干进程之间 形成一种头尾相接的循环等待资源关系。
如何解决:
(1)资源一次性分配,从而解决请求保持的问题
(2)可剥夺资源:当进程 新的资源未得到满足时,释放已有的资源;
(3)资源有序分配: 资源按序号递增,进程请求按递增 请求,释放则相反。
答案解析
举个例子,比如:如果此时有两个线程T1和T2,它们分别占有R1和R2资源
此时,T1请求R2资源的同时,T2请求R1资源。
这个时候T2说:你把R1给我,我就给你R2
T1说:不行,你要先给我R2,我才能给你R1
那么就这样,死锁产生了。如下图:
image-20220713161919466
2.40 有了进程,为什么还要有线程?
参考回答
进程在早期的多任务操作系统中是基本的执行单元。每次进程切换,都要先保存进程资源然后再恢
复,这称为上下文切换。但是进程频繁切换将引起额外开销,从而严重影响系统的性能。为了减少
进程切换的开销,人们把两个任务放到一个进程中,每个任务用一个更小粒度的执行单元来实现并
发执行,这就是线程。
- 线程与进程对比
(1)进程间的信息难以共享。由于除去 只读代码段外, 父子进程并未共享内存,因此必须采用一
些进程间通信方式,在进程间进行信息交换。
但多个线程共享进程的内存,如代码段、数据段、扩展段,线程间进行信息交换十分方便。
(2) 调用 fork() 来创建进程的代价相对较高,即便利用写时复制技术,仍然需要复制诸如内存页
表和文件描述符表之类的多种进程属性,这意味着 fork() 调用在时间上的开销依然不菲。
但创建线程比创建进程通常要快 10 倍甚至更多。线程间是 共享虚拟地址空间的,无需采用写时复
制来复制内存,也无需复制页表。
2.41 单核机器上 写 多线程程序,是否要考虑加锁,为什么?
参考回答
在单核机器上写多线程程序,仍然需要线程锁。
原因: 因为 线程锁 通常用来实现线程的同步和通信。在单核机器上的多线程程序, 仍然存在线程同步的
问题。因为在抢占式操作系统中,通常为每个线程分配一个时间片,当某个线程时间片耗尽时,操作系
统会将其挂起,然后运行另一个线程。如果这两个线程共享某些数据,不使用线程锁的前提下,可能会
导致共享数据修改引起冲突。
2.42 说说多线程和多进程的不同?
参考回答:
(1)一个线程 从属于 一个进程;一个进程 可以包含多个线程。
(2)一个 线程挂掉,对应的进程挂掉, 多线程也挂掉; 一个进程挂掉,不会影响其他进程,多进程稳
定。
(3)进程系统开销 显著大于 线程开销; 线程需要的系统资源更少。
(4)多个进程 在执行时拥有各自独立的内存单元, 多个线程 共享进程的内存,如代码段、数据段、扩
展段;但每个线程拥有自己的栈段和寄存器组。
(5)多进程切换时 需要刷新TLB并获取新的地址空间,然后切换硬件上下文和内核栈; 多线程切换时只
需要切换 硬件上下文和内核栈。
(6)通信方式不一样。
(7)多进程适应于多核、多机分布;多线程适用于多核
2.43 简述互斥锁的机制,互斥锁与读写的区别?
参考回答
互斥锁机制: mutex,用于保证在任何时刻,都只能有一个线程访问该对象。当获取 锁操作失败
时,线程会进入睡眠,等待锁释放时被唤醒。
互斥锁和读写锁:
(1) 读写锁 区分 读者和写者,而 互斥锁不区分
(2)互斥锁 同一时间只允许一个线程访问该对象,无论读写; 读写锁同一时间内只允许一个写
者,但是允许多个读者同时读对象。
答案解析:
原理详解:
互斥锁其实就是一个bool型变量,为true时表示锁可获取,为false时表示已上锁 。这里说的是互斥锁,
其实是泛指linux中所有的锁机制。
我们采用 互斥锁保护临界区,从而防止竞争条件。也就是说,一个线程在进入临界区时应得到锁;它在
退出临界区时释放锁。函数 acquire() 获取锁,而函数 release() 释放锁,如图 :
image-20220713165031605
每个互斥锁有一个布尔变量 available,它的值表示锁是否可用。如果锁是可用的,那么调用 acquire()
会成功,并且锁不再可用。当一个线程试图获取不可用的锁时,它会阻塞,直到锁被释放。
按如下定义 acquire():
acquire() {
while (!available);
/* busy wait */
available = false;
}
按如下定义release():
release() {
available = true;
}
2.44 说说什么是信号量,有什么作用?
参考回答
概念:信号量本质上是一个计数器, 用于 多进程对共享数据对象的读取, 它主要是用来保护共享资
源(信号量也属于临界资源),使得资源在一个时刻只有一个进程独享。
原理:由于信号量只能进行两种操作等待和发送信号,即P(sv)和V(sv),具体的行为如下:
(1)P(sv)操作: 如果sv的值大于零,就给它减1;如果它的值为零,就挂起该进程的执行(信号
量的值为正,进程获得该资源的使用权, 进程将信号量减1,表示它使用了一个资源单位)。
(2)V(sv)操作: 如果 有其他进程因 等待sv而被挂起,就让它恢复运行,如果没有进程因等待sv而
挂起,就给它加1(若此时信号量的值为0,则进程进入挂起状态,直到信号量的值大于0,若进程
被唤醒则返回至第一步)。
作用:用于多进程对共享数据对象的读取,它主要是用来保护共享资源(信号量也属于临界资
源),使得资源在一个时刻只有一个进程独享。
2.45 进程、线程的中断切换的过程是怎样的?
参考回答
上下文切换指的是内核(操作系统的核心)在CPU上对进程或者线程进行切换。
进程上下文切换
(1)保护 被中断进程的 处理器现场信息
(2)修改 被中断进程的 进程控制块有关信息,如进程状态等
(3)把 被中断进程的进程控制块 加入 有关队列
(4)选择 下一个占有处理器运行的进程
(5)根据 被选中进程设置操作系统用到的地址转换和存储保护信息
切换页目录以使用新的地址空间
切换内核栈和硬件上下文(包括分配的内存,数据段,堆栈段等)
(6)根据 被选中进程 恢复 处理器现场
线程上下文切换
(1)保护 被中断线程的 处理器现场信息
(2)修改 被中断线程的 线程控制块有关信息,如线程状态等
(3)把 被中断线程的线程控制块 加入有关队列
(4)选择 下一个占有处理器运行的 线程
(5)根据 被选中线程设置操作系统用到的 存储保护信息
切换内核栈和硬件上下文(切换堆栈,以及各寄存器)
(6)根据 被选中线程 恢复处理器现场
2.46 简述自旋锁和互斥锁的使用场景
参考回答
互斥锁用于 临界区持锁时间比较长的操作,比如下面这些情况都可以考虑
(1)临界区 有IO操作
(2)临界区 代码复杂或者循环量大
(3)临界区 竞争非常激烈
(4)单核处理器
自旋锁就主要用在 临界区持锁时间非常短且CPU资源不紧张的情况下。
2.47 请你说说线程有哪些状态,相互之间怎么转换?
参考回答
类似进程,有以下五种状态:
新建状态(New)
就绪状态(Runnable)
运行状态(Running)
阻塞状态(Blocked)
死亡状态(Dead)
转换方式如下:
创建状态
一个应用程序从系统上启动,首先就是进入创建状态,获取系统资源。
就绪状态
在创建状态完成之后,线程已经准备好,处于就绪状态,但是还未获得处理器资源,无法运行。
运行状态
获取处理器资源,被系统调度,当具有时间片开始进入运行状态。如果线程的时间片用完了就进入就绪
状态。
阻塞状态
在运行状态期间,如果进行了阻塞的操作,如耗时的I/O操作,此时线程暂时无法操作就进入到了阻塞
状态,在这些操作完成后就进入就绪状态。等待再次获取处理器资源,被系统调度,当具有时间片就进
入运行状态。
终止状态
线程结束或者被系统终止,进入终止状态
相互转换如图:
image-20220713170333181
2.48 多线程和单线程有什么区别,多线程编程要注意什么,多线程
加锁需要注意什么?
参考回答
区别:
(1) 多线程 从属于 一个进程,单线程 也从属于 一个进程;一个线程 挂掉都会导致从属的 进程挂
掉。
(2)一个进程里有多个线程,可以并发执行多个任务;一个进程里只有一个线程,就只能执行一
个任务。
(3) 多线程 并发执行 多任务,需要切换内核栈与硬件上下文,有切换的开销;单线程 不需要切
换,没有切换的开销。
(4)多线程 并发执行多任务,需要考虑同步的问题;单线程不需要考虑同步的问题。
多线程编程需要考虑同步的问题。线程间的同步方式包括互斥锁、信号量、条件变量、读写锁。
多线程加锁,主要需要注意死锁的问题。破坏死锁的必要条件从而避免死锁。
答案解析
答案解析
死锁: 是指多个进程在执行过程中,因争夺资源而造成了互相等待。此时系统产生了死锁。比如两
只羊过独木桥,若两只羊互不相让,争着过桥,就产生死锁。
产生的条件:死锁发生有四个必要条件:
(1)互斥条件:进程对所分配到的资源不允许其他进程访问,若其他进程访问,只能等待,直到
进程使用完成后释放该资源;
(2)请求保持条件:进程获得一定资源后,又对其他资源发出请求,但该资源被其他进程占有,
此时请求阻塞,而且该进程不会释放自己已经占有的资源;
(3)不可剥夺条件:进程已获得的资源,只能自己释放,不可剥夺;
(4)环路等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
如何解决:
(1)资源一次性分配,从而解决请求保持的问题
(2)可剥夺资源:当进程新的资源未得到满足时,释放已有的资源;
(3)资源有序分配:资源按序号递增,进程请求按递增请求,释放则相反。
答案解析
举个例子,比如:如果此时有两个线程T1和T2,它们分别占有R1和R2资源
此时,T1请求R2资源的同时,T2请求R1资源。
这个时候T2说:你把R1给我,我就给你R2
T1说:不行,你要先给我R2,我才能给你R1
那么就这样,死锁产生了。如下图:a
image-20220713170702387
2.49 说说sleep和wait的区别?
参考回答
sleep
sleep是一个延时函数,让进程或线程进入休眠。休眠完毕后继续运行。
在linux下面,sleep函数的参数是秒,而windows下面sleep的函数参数是毫秒。
windows下面sleep的函数参数是毫秒。
#include <windows.h>// 首先应该先导入头文件
Sleep (500) ; //注意第一个字母是大写。
//就是到这里停半秒,然后继续向下执行。
在 Linux C语言中 sleep的单位是秒
#include <unistd.h>// 首先应该先导入头文件
sleep(5); //停5秒
//就是到这里停5秒,然后继续向下执行。
wait
wait是父进程回收子进程PCB资源的一个系统调用。进程 一旦调用了wait函数,就立即阻塞自己本
身,然后由wait函数 自动分析当前进程的某个子进程 是否 已经退出,当找到一个已经变成僵尸的子
进程,wait就会收集这个子进程的信息,并把它彻底销毁后返回; 如果没有找到这样一个子进程,
wait就会一直阻塞,直到有一个出现为止。函数原型如下:
#include<sys/types.h>
#include<sys/wait.h>
pid_t wait(int* status);
子进程的结束状态值会由参数status返回,而子进程的进程识别码也会一起返回。如果不需要结束
状态值,则参数status可以设成 NULL。
区别:
(1)sleep是一个延时函数,让进程或线程进入休眠。休眠完毕后继续运行。
(2)wait是父进程回收子进程PCB(Process Control Block)资源的一个系统调用。
2.50 说说线程池的设计思路,线程池中线程的数量由什么确定?
参考回答
设计思路:
实现线程池有以下几个步骤:
(1)设置一个 生产者消费者队列,作为临界资源。
(2)初始化 n个线程,并让其运行起来,加锁 去队列里取任务运行
(3)当 任务队列为空 时,所有线程阻塞。
(4)当生产者队列来了一个任务后,先对队列加锁,把任务挂到队列上,然后使用条件变量去通
知阻塞中的一个线程来处理。
- 线程池中线程数量:
线程数量和哪些因素有关:CPU,IO、并行、并发
如果是 CPU密集型应用,则线程池大小设置为: CPU数目+1
如果是 IO密集型应用,则线程池大小设置为: 2*CPU数目+1
最佳线程数目 = (线程等待时间 与 线程CPU时间之比 + 1)* CPU数目
所 以线程等待时间 所占比例 越高,需要越多线程。线程CPU时间所占比例越高,需要 越少线程。
答案解析
为什么要创建线程池:
创建线程和销毁线程的花销是比较大的,这些时间 有可能比处理业务的时间还要长。这样 频繁的创
建线程和销毁线程,再加上业务工作线程,消耗系统资源的时间,可能导致系统资源不足。同时线
程池也是为了提升系统效率。
线程池的核心线程与普通线程:
任务队列可以存放100个任务, 此时为空, 线程池里有10个核心线程,若突然来了10个任务,那
么刚好10个核心线程直接处理; 若又来了90个任务,此时核心线程来不及处理,那么有80个任务
先入队列,再创建核心线程处理任务; 若又来了120个任务,此时任务队列已满,不得已,就得创
建20个普通线程来处理多余的任务。
以上是线程池的工作流程。
2.51 进程和线程相比,为什么慢?
参考回答s
进程系统开销 显著大于 线程开销; 线程需要的系统资源更少。
进程切换 开销比 线程大。多进程切换时需要 刷新TLB并获取新的地址空间,然后切换 硬件上下文和
内核栈; 多线程切换时只需要切换硬件上下文和内核栈。
进程通信 比 线程通信开销大。进程通信需要借助管道、队列、共享内存,需要额外申请空间,通信
繁琐; 而线程共享 进程的内存,如代码段、数据段、扩展段,通信快捷简单,同步开销更小。
2.52 简述Linux零拷贝的原理?
参考回答:
1. 什么是零拷贝:
所谓「零拷贝」描述的是计算机操作系统当中,CPU不执行将 数据从 一个内存区域,拷贝到 另外一
个内存区域的任务。通过网络传输文件时,这样通常可以节省 CPU 周期和内存带宽。
零拷贝的好处:
(1)节省了 CPU 周期,空出的 CPU 可以完成更多其他的任务
(2)减少了内存区域之间数据拷贝,节省内存带宽
(3)减少用户态和内核态之间数据拷贝,提升数据传输效率
(4)应用零拷贝技术,减少用户态和内核态之间的上下文切换
- 零拷贝原理:
在传统 IO 中,用户态空间与内核态空间之间的复制是完全不必要的,因为用户态空间仅仅起到了
一种数据转存媒介的作用,除此之外没有做任何事情。
(1)Linux 提供了 sendfile() 用来减少我们的数据拷贝和上下文切换次数。
image-20220714103955832
a. 发起 sendfile() 系统调用,操作系统由用户态空间切换到内核态空间(第一次上下文切换)
b. 通过 DMA 引擎将数据从磁盘拷贝到内核态空间的输入的 socket 缓冲区中(第一次拷贝)
c. 将数据从内核空间拷贝到与之关联的 socket 缓冲区(第二次拷贝)
d. 将 socket 缓冲区的数据拷贝到协议引擎中(第三次拷贝)
e. sendfile() 系统调用结束,操作系统由用户态空间切换到内核态空间(第二次上下文切换)
根据以上过程,一共有 2 次的上下文切换,3 次的 I/O 拷贝。我们看到从用户空间到内核空间并没
有出现数据拷贝,从操作系统角度来看,这个就是零拷贝。内核空间出现了复制的原因: 通常的硬
件在通过DMA访问时期望的是连续的内存空间。
(2)mmap 数据零拷贝原理
如果需要对数据做操作,Linux 提供了mmap 零拷贝来实现。
2.53 简述epoll和select的区别,epoll为什么高效?
参考回:答
(1)每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销 在fd很多时会很大;而
epoll保证了每个fd在整个过程中只会拷贝一次。
(2)每次调用select都需要在内核遍历传递进来的所有fd; 而epoll只需要轮询一次fd集合,同时
查看就绪链表中有没有就绪的fd就可以了。
(3)select支持的文件描述符数量太小了,默认是1024;而epoll没有这个限制,它所支持的fd上
限是最大可以打开文件的数目,这个数字一般远大于2048。
- epoll为什么高效:
(1)select,poll实现需要自己不断轮询所有fd集合,直到设备就绪,期间可能要睡眠和唤醒多次
交替。而epoll只要判断一下就绪链表是否为空就行了,这节省了大量的CPU时间。
(2)select,poll每次调用 都要把fd集合从用户态往内核态拷贝一次,并且要把当前进程往设备等
待队列中挂一次, 而epoll只要一次拷贝,而且把当前进程往等待队列上挂也只挂一次,这也能节
省不少的开销。
2.54 说说多路IO复用技术有哪些,区别是什么?
参考回答:
select,poll,epoll都是IO多路复用的机制,I/O多路复用就是通过一种机制,可以监视多个文件
描述符,一旦某个文件描述符就绪(一般是读就绪或者写就绪),能够通知应用程序进行相应的读
写操作。
区别:
(1)poll与select不同, 通过一个pollfd数组向内核传递需要关注的事件, 故没有描述符个数的限
制,pollfd中的events字段和revents分别用于标示关注的事件和发生的事件,故pollfd数组只需要
被初始化一次。
(2)select,poll实现需要自己不断轮询所有fd集合,直到设备就绪,期间可能要睡眠和唤醒多次
交替。而epoll只要判断一下就绪链表是否为空就行了,这节省了大量的CPU时间。
(3)select,poll每次调用 都要把fd集合从用户态往内核态拷贝一次,并且要把当前进程往设备等
待队列中挂一次,而epoll只要一次拷贝,而且把当前进程往等待队列上挂也只挂一次,这也能节
省不少的开销。
2.55 简述socket中select,epoll的使用场景和区别,epoll水平触
发与边缘触发的区别?
参考回答:
select,epoll的使用场景:都是IO多路复用的机制,应用于高并发的网络编程的场景。I/O多路复
用就是通过一种机制,可以监视多个文件描述符,一旦某个文件描述符就绪(一般是读就绪或者写
就绪),能够通知应用程序进行相应的读写操作。
select,epoll的区别:
(1)每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时会很大;而
epoll保证了每个fd在整个过程中只会拷贝一次。
(2)每次调用select都需要在内核遍历传递进来的所有fd;而epoll只需要轮询一次fd集合,同时
查看就绪链表中有没有就绪的fd就可以了。
(3)select支持的文件描述符数量太小了,默认是1024;而epoll没有这个限制,它所支持的fd上
限是最大可以打开文件的数目,这个数字一般远大于2048。
epoll水平触发与边缘触发的区别
LT模式(水平触发)下,只要这个fd还有数据可读,每次 epoll_wait都会返回它的事件,提醒用户
程序去操作;
而在ET(边缘触发)模式中,它只会提示一次,直到下次再有数据流入之前 都不会再提示了,无
论fd中是否还有数据可读。
2.56 说说Reactor、Proactor模式。
参考回答
在高性能的I/O设计中,有两个比较著名的模式Reactor和Proactor模式,其中Reactor模式用于同步
I/O,而 Proactor运用于异步I/O操作。
Reactor模式:Reactor模式应用于 同步I/O 的场景。Reactor中读操作的具体步骤如下:
读取操作:
(1)应用程序 注册 读就需 事件和相关联的事件处理器
(2)事件分离器 等待事件的发生
(3)当发生 读就需事件的时候,事件分离器调用第一步注册的事件处理器
(4)事件处理器 首先执行实际的读取操作,然后根据读取到的内容进行进一步的处理
Proactor模式:Proactor模式应用于异步I/O的场景。Proactor中读操作的具体步骤如下:
(1)应用程序 初始化一个 异步读取操作 ,然后注册相应的事件处理器,此时 事件处理器 不关注读
取就绪事件,而是关注读取完成事件,这是区别于Reactor的关键。
(2)事件分离器 等待 读取操作完成事件
(3)在事件分离器 等待 读取操作完成的时候, 操作系统 调用内核线程完成读取操作,并将读取的
内容 放入 用户传递过来的缓存区中。这也是区别于Reactor的一点, Proactor中,应用程序需要传
递缓存区。
(4)事件分离器捕获到读取完成事件后, 激活 应用程序注册的事件处理器,事件处理器直接从缓
**存区读取数据,而不需要进行实际的读取操作。
区别:
从上面可以看出,Reactor中需要应用程序自己读取或者写入数据, 而Proactor模式中,应
用程序不需要用户再自己接收数据,直接使用就可以了,操作系统会将数据从内核拷贝到用户区。
答案解析
IO模型的类型。
(1)阻塞IO:调用者调用了某个函数,等待这个函数返回,期间什么也不做,不停的检查这个函数有
没有返回,必须等这个函数返回后才能进行下一步动作。
(2)非阻塞IO:非阻塞等待,每隔一段时间就去检查IO事件是否就绪。没有就绪就可以做其他事情。
(3)信号驱动IO:Linux用 套接口进行信号驱动IO,安装一个信号处理函数,进程继续运行并不阻塞,
当IO事件就绪,进程收到SIGIO信号,然后处理IO事件。
(4)IO多路复用:Linux用select/poll函数实现IO复用模型,这两个函数也会使进程阻塞,但是和阻塞
IO所不同的是这两个函数可以同时阻塞多个IO操作。而且可以同时对多个读操作、写操作的IO函数进行
检查。知道有数据可读或可写时,才真正调用IO操作函数。
(5)异步IO:Linux中,可以调用aio_read函数告诉内核描述字缓冲区指针和缓冲区的大小、文件偏移
及通知的方式,然后立即返回,当内核将数据拷贝到缓冲区后,再通知应用程序。用户可以直接去使用
数据。
前四种模型–阻塞IO、非阻塞IO、多路复用IO和信号驱动IO都属于同步模式,因为其中真正的IO操作(函
数)都将会阻塞进程,只有异步IO模型真正实现了IO操作的异步性。
2.57 简述同步与异步的区别,阻塞与非阻塞的区别?
同步与异步的区别:
同步:是所有的操作都做完,才返回给用户结果。即写完数据库之后,再响应用户,用户体验不
好。
异步:不用等所有操作都做完,就响应用户请求。即先响应用户请求,然后慢慢去写数据库,用户
体验较好。
- 阻塞与非阻塞的区别:
阻塞:调用者 调用了某个函数,等待这个函数返回,期间什么也不做,不停的检查这个函数有没有
返回,必须等这个函数返回后才能进行下一步动作。
非阻塞:非阻塞等待 ,每隔一段时间就去检查IO事件是否就绪。没有就绪就可以做其他事情。
2.58 BIO、NIO有什么区别?
参考回答
BIO(Blocking I/O):阻塞IO。调用者调用了某个函数,等待这个函数返回,期间什么也不做,不停
的检查这个函数有没有返回,必须等这个函数返回后才能进行下一步动作。
NIO(New I/O):同时支持阻塞与非阻塞模式,NIO的做法是叫一个线程不断的轮询每个IO的状态,
看看是否有IO的状态发生了改变,从而进行下一步的操作。
2.59 请介绍一下5种IO模型
参考回答:
阻塞IO:调用者调用了某个函数,等待这个函数返回,期间什么也不做,不停的检查这个函数有
没有返回,必须等这个函数返回后才能进行下一步动作。
非阻塞IO:非阻塞等待,每隔一段时间就去检查IO事件是否就绪。没有就绪就可以做其他事情。
信号驱动IO:Linux用套接口进行信号驱动IO,安装一个信号处理函数,进程继续运行并不阻塞,
当IO事件就绪,进程收到SIGIO信号,然后处理IO事件。
IO多路复用:Linux用select/poll函数实现IO复用模型,这两个函数也会使进程阻塞,但是和阻塞
IO所不同的是这两个函数可以同时阻塞多个IO操作。而且可以同时对多个读操作、写操作的IO函数进行检查。知道有数据可读或可写时,才真正调用IO操作函数。
异步IO:Linux中,可以调用aio_read函数告诉内核描述字缓冲区指针和缓冲区的大小、文件偏移
及通知的方式,然后立即返回,当内核将数据拷贝到缓冲区后,再通知应用程序。用户可以直接去
使用数据。
答案解析:
前四种模型–阻塞IO、非阻塞IO、多路复用IO和信号驱动IO都属于同步模式,因为其中真正的IO操作(函
数)都将会阻塞进程,只有异步IO模型真正实现了IO操作的异步性。
异步和同步的区别就在于,异步是 内核将数据拷贝到用户区,不需要用户再自己接收数据,直接使用就
可以了,而 同步是内核通知用户数据到了,然后用户自己调用相应函数去接收数据。
2.60 请说一下socket网络编程中客户端和服务端用到哪些函数?
参考回答
服务器端函数:
(1)socket创建一个套接字
(2)bind绑定ip和port
(3)listen使套接字变为可以被动链接
(4)accept等待客户端的链接
(5)write/read接收发送数据
(6)close关闭连接
客户端函数:
(1)创建一个socket,用函数socket()
(2)bind绑定ip和port
(3)连接服务器,用函数connect()
(4)收发数据,用函数send()和recv(),或read()和write()
(5)close关闭连接
image-20220714111107917
2.61 简述网络七层参考模型,每一层的作用?
image-20220714111201478
image-20220714111226587
3.计算机网络
3.1 简述静态路由和动态路由
参考回答
静态路由是由 系统管理员设计与构建的路由表规定的路由。适用于网关数量有限的场合,且网络拓
朴结构不经常变化的网络。其缺点 是不能动态地适用网络状况的变化,当网络状况变化后必须由网
络管理员修改路由表。
动态路由 是由 路由选择协议而动态构建的,路由协议 之间 通过交换各自所拥有的路由信息实时更新
路由表的内容。动态路由可以 自动学习网络的拓朴结构,并更新路由表。其缺点是 路由广播更新信
息将占据大量的网络带宽
3.2 说说有哪些路由协议,都是如何更新的
参考回答:’
路由可分为静态&动态路由。静态路由由 管理员手动维护;动态路由 由路由协议自动维护。
路由选择算法 的必要步骤:
1)向 其它路由器 传递路由信息;
2)接收 其它路由器 的路由信息;
3)根据 收到的路由信息 计算出到每个目的网络的最优路径,并由此生成路由选择表;
4)根据 网络拓扑的变化 及时的做出反应,调整路由生成新的路由选择表,同时把拓扑变化以路由
信息的形式向其它路由器宣告。
两种主要算法:距离向量法(Distance Vector Routing)和链路状态算法(Link-State
Routing)
由此可分为距离矢量(如:RIP、IGRP、EIGRP)& 链路状态路由协议(如:OSPF、IS-IS)。 路由
协议是路由器之间实现路由信息共享的一种机制,它允许 路由器之间相互交换和维护各 自的路由
表。当一台路由器的路由表由于某种原因发生变化时,它需要及时地将这一变化通 知与之相连接
的其他路由器,以保证数据的正确传递。路由协议 不承担网络上终端用户之间 的数据传输任务。
1)RIP 路由协议:
RIP 协议最初是为 Xerox 网络系统的 Xerox parc 通用协议而设计的,是
Internet 中常用的 路由协议。RIP 采用距离向量算法,即路由器根据距离选择路由,所以也称为
距离向量协议。 路由器收集所有可到达目的地的不同路径,并且保存有关到达每个目的地的最少
站点数的路 径信息,除到达目的地的最佳路径外,任何其它信息均予以丢弃。同时路由器也把所收集的 路由信息用 RIP 协议通知相邻的其它路由器。这样,正确的路由信息逐渐扩散到了全网。
RIP 使用非常广泛,它简单、可靠,便于配置。但是 RIP 只适用于小型的同构网络,因 为它允许
的最大站点数为 15,任何超过 15 个站点的目的地均被标记为不可达。而且 RIP 每 隔 30s 一次的
路由信息广播也是造成网络的广播风暴的重要原因之一。
2)OSPF 路由协议:
0SPF 是一种基于链路状态的路由协议,需要每个路由器向其同一管理域的所
有其它路 由器发送链路状态广播信息。在 OSPF 的链路状态广播中包括所有接口信息、所有的量
度和 其它一些变量。利用 0SPF 的路由器首先必须收集有关的链路状态信息,并根据一定的算法
计算出到每个节点的最短路径。而基于距离向量的路由协议仅向其邻接路由器发送有关路由 更新
信息。与 RIP 不同,OSPF 将一个自治域再划分为区,相应地即有两种类型的路由选择方式: 当源
和目的地在同一区时,采用区内路由选择;当源和目的地在不同区时,则采用区间路由 选择。这
就大大减少了网络开销,并增加了网络的稳定性。当一个区内的路由器出了故障时 并不影响自治
域内其它区路由器的正常工作,这也给网络的管理、维护带来方便。
3)BGP 和 BGP4 路由协议:
BGP 是为 TCP/IP 互联网设计的外部网关协议,用于多个自治域之
间。它既不是基于纯 粹的链路状态算法,也不是基于纯粹的距离向量算法。它的主要功能是与其
它自治域的 BGP 交换网络可达信息。各个自治域可以运行不同的内部网关协议。BGP 更新信息包
括网络号/ 自治域路径的成对信息。自治域路径包括到达某个特定网络须经过的自治域串,这些
更新信 息通过 TCP 传送出去,以保证传输的可靠性。为了满足 Internet 日益扩大的需要,BGP
还在不断地发展。在最新的 BGP4 中,还可以 将相似路由合并为一条路由。
4)IGRP 和 EIGRP 协议:EIGRP 和早期的 IGRP 协议都是由 Cisco 发明,是基于距离向量算法的
动态路由协议。 EIGRP(Enhanced Interior Gateway Routing Protocol)是增强版的 IGRP 协议。
它属于动态内部网 关路由协议,仍然使用矢量-距离算法。但它的实现比 IGRP 已经有很大改
进,其收敛特性 和操作效率比 IGRP 有显著的提高。它的收敛特性是基于 DUAL ( Distributed
Update Algorithm ) 算法的。DUAL 算法使得路径 在路由计算中根本不可能形成环路。它的收敛
时间可以与已存在的其他任何路由协议相匹敌
Enhanced IGRP 与其它路由选择协议之间主要区别包括:收敛宽速(Fast Convergence)、 支
持变长子网掩模(Subnet Mask)、局部更新和多网络层协议。执行 Enhanced IGRP 的路由 器存
储了所有其相邻路由表,以便于它能快速利用各种选择路径(Alternate Routes)。如果没有合适
路径,Enhanced IGRP 查询其邻居以获取所需路径。直到找到合适路径,EnhancedIGRP 查询才
会终止,否则一直持续下去。
EIGRP 不作周期性更新。取而代之,当路径度量标准改变时,Enhanced IGRP 只发送局 部更新
(Partial Updates)信息。局部更新信息的传输自动受到限制,从而使得只有那些需 要信息的路
由器才会更新。基于以上这两种性能,因此 Enhanced IGRP 损耗的带宽比 IGRP 少得多。
3.3 简述域名解析过程,本机如何干预域名解析
参考回答
(1)在浏览器中输入 www.qq 域名,
操作系统会先检查自己本地的hosts文件是否有这个网址
映射关系,如果有,就先调用这个IP地址映射,完成域名解析。
(2)如果hosts里没有这个域名的映射,则 查找本地DNS解析器缓存,是否有这个网址映射关
系,如果有,直接返回,完成域名解析。
(3)如果hosts与本地DNS解析器缓存 都 没有相应的网址映射关系,首先会找TCP/IP参数中设置
的首选DNS服务器,在此我们叫它本地DNS服务器,此服务器收到查询时,如果要查询的域名,
包含在本地配置区域资源中,则返回解析结果给客户机,完成域名解析,此解析具有权威性。
(4)如果要查询的域名,不由本地DNS服务器区域解析,但该服务器已缓存了此网址映射关系,
则调用这个IP地址映射,完成域名解析, 此解析不具有权威性。
(5)如果本地DNS服务器本地区域文件与缓存解析都失效,则 根据本地DNS服务器的设置(是否
设置转发器)进行查询,如果未用转发模式,本地DNS就把请求发至13台根DNS,根DNS服务器
收到请求后会判断这个域名()是谁来授权管理,并会返回一个负责该顶级域名服务器的一个
IP。
本地DNS服务器 收到IP信息后,将会联系负责域的这台服务器。这台负责域的服务
器 收到请求后, 如果自己无法解析,它就会找一个管理域的下一级DNS服务器地址(qq)
给本地DNS服务器。当本地DNS服务器收到这个地址后,就会找qq域服务器,重复上面的动
作,进行查询,直至找到 www.qq 主机。
(6)如果用的是转发模式,此DNS服务器就会把请求转发至上一级DNS服务器,由上一级服务器
进行解析,上一级服务器如果不能解析,或 找根DNS或把转请求转至上上级,以此循环。不管是
本地DNS服务器用是是转发,还是根提示,最后都是把结果返回给本地DNS服务器,由此DNS服
务器再返回给客户机。
从客户端到本地DNS服务器是属于递归查询,而 DNS服务器之间就是的交互查询就是迭代查询。
通过修改本机host来干预域名解析,例如:
在/etc/hosts文件中添加一句话
192.168.188.1 www.baidu
保存文件后再ping一下www.baidu就会连接到192.168.188.1
每一行为一条记录,分成两部分,第一部分是IP,第二部分是域名。
一个IP后面可以跟多个域名,可以是几十个甚至上百个
每一行只能有一个IP,也就是说一个域名不能对应多个IP
如果有多行中出现相同的域名(对应的ip不一样),会按最前面的记录来解析
3.4 简述 DNS 查询服务器的基本流程是什么?DNS 劫持是什么?
参考回答:
打开浏览器,输入一个域名。比如输入www.163,这时,你使用的电脑会发出一个DNS请求
到本地DNS服务器。本地DNS服务器一般都是你的网络接入服务器商提供,比如中国电信,中国
移动。
DNS请求到达本地DNS服务器之后, 本地DNS服务器会首先查询它的缓存记录,如果缓存中有此
条记录,就可以直接返回结果。 如果没有,本地DNS服务器还要向DNS根服务器进行查询。
根DNS服务 器没有记录具体的域名和IP地址的对应关系,而是告诉 本地DNS服务器,你可以到 域服
务器上去继续查询,并给出域服务器的地址。
本地DNS服务器继续向域服务器发出请求,在这个例子中,请求的对象是域服务器。域
服务器收到请求之后,也 不会直接返回域名和IP地址的对应关系,而是 告诉本地DNS服务器,你的
域名的解析服务器的地址。
最后,本地DNS服务器向 域名的解析服务器发出请求,这时就能收到一个域名和IP地址对应关系,
本地DNS服务器不仅要把IP地址返回给用户电脑,还要把这个对应关系保存在缓存中,以备下次别
的用户查询时,可以直接返回结果,加快网络访问。
DNS劫持就是 通过劫持 了DNS服务器,通过某些手段取得某域名的解析记录控制权,进而修改此
域名的解析结果,导致对该域名的访问 由原IP地址 转入到修改后的指定IP,其结果就是对特定的网
址 不能访问或访问的是假网址,从而实现窃取资料或者破坏原有正常服务的目的。DNS劫持 通过
篡改DNS服务器上的数据返回给用户一个错误的查询结果来实现的。
DNS劫持症状: 在某些地区的用户在成功连接宽带后,首次打开任何页面都指向ISP提供的“电信互
联星空”、“网通黄页广告”等内容页面。还有就是曾经出现过 用户 访问Google域名的时候出现了百
度的网站。这些都属于DNS劫持。
3.5 简述网关的作用是什么,同一网段的主机如何通信
参考回答
网关即网络中的关卡,我们的互联网是一个一个的局域网、城域网、等连接起来的,在连接点上就
是一个一个网络的关卡,即我们的网关,他是保证网络互连的,翻译和转换,使得不同的网络体系
能够进行。
网内通信,即通信双方都位处同一网段中,数据传输无需经过路由器(或三层交换机),即可由本网
段自主完成。
假设发送主机的 ARP表 中 并无目的主机对应的表项,则发送主机会以目的主机IP地址为内容,广播
ARP请求 以期获知目的主机MAC地址,并通过交换机(除到达端口之外的所有端口发送,即洪泛
(Flooding))向全网段主机转发,而只有目的主机接收到此ARP请求后会将自己的MAC地址和IP地址
装入ARP应答后将其回复给发送主机,发送主机 接收到此ARP应答后,从中提取 目的主机的MAC地
址,并在其ARP表中建立目的主机的对应表项(IP地址到MAC地址的映射),之后即可向目的主机发
送数据,将待发送数据封装成帧,并通过二层设备(如交换机)转发至本网段内的目的主机,自此完
成通信。
3.6 简述CSRF攻击的思想以及解决方法
参考回答
CSRF全称叫做,跨站请求伪造。就是 黑客可以伪造用户的身份去做一些操作,进而满足自身目
的。
要完成一次CSRF攻击,受害者必须依次完成两个步骤:
1)登录受信任网站A,并在本地生成Cookie。
2)在不登出A的情况下,访问危险网站B。
此时,黑客就可以获取你的cookie达成不可告人的目的了。
CSRF 攻击是一种请求伪造的攻击方式,它利用的是服务器不能识别用户的类型从而盗取用户的信
息来攻击。因此要防御该种攻击,因为 从服务器端着手,增强服务器的识别能力,设计良好的防御
机制。主要有以下几种方式:
1)请求头中的Referer验证(不推荐)
****HTTP的头部有一个 Referer 信息的字段,它记录着该次HTTP请求的来源地址(即它从哪里来
的),既然CSRF攻击是伪造请求 是从服务器发送过来的,那么 我们就禁止跨域访问,在 服务器端增
加验证,过滤掉那些不是从本服务器发出的请求,这样可以在一定程度上避免CSRF攻击。 但是这
也有缺点,比如如果是从搜索引擎所搜结果调整过来,请求也会被认为是跨域请求。
2)请求令牌验证(token验证)
token验证是一种比较广泛使用的防止 CSRF攻击 的手段, 当用户通过正常渠道访问服务器时,服
务器会生成一个随机的字符串保存在session中, 并作为令牌(token)返回给客户端,以隐藏的
形式保存在客户端中,客户端每次请求都会带着这个token,服务器根据该token判断该请求是否
合法
3.7 说说 MAC地址和IP地址分别有什么作用
IP地址 是 IP协议 提供的一种统一的地址格式,它为互联网上的每一个网络和每一台主机分配一个逻
辑地址, 以此来屏蔽物理地址的差异。而 MAC地址,指的是物理地址,用来定义网络设备的位
置。
IP地址的分配 是根据网络的拓扑结构,而不是根据谁制造了网络设置。若将高效的路由选择方案建
立在设备制造商的基础上而不是网络所处的拓朴位置基础上,这种方案是不可行的。
当存在一个附加层的地址寻址时,设备更易于移动和维修。例如,如果一个以太网卡坏了,可以被
更换,而无须取得一个新的IP地址。如果一个IP主机从一个网络移到另一个网络,可以给它一个新
的IP地址,而无须换一个新的网卡。
无论是局域网,还是广域网中的计算机之间的通信,最终都表现为将数据包从某种形式的链路上的
初始节点出发,从一个节点传递到另一个节点,最终传送到目的节点。数据包在这些节点之间的移
动都是由ARP(Address Resolution Protocol:地址解析协议)负责将IP地址映射到MAC地址上
来完成的。
3.8 简述 TCP 三次握手和四次挥手的过程
image-20220714142050321
1)第一次握手:建立连接时,客户端向服务器发送SYN包(seq=x),请求建立连接,等待确认
2)第二次握手:服务端收到客户端的SYN包,回一个ACK包(ACK=x+1)确认收到,同时发送一个SYN
包(seq=y)给客户端
3)第三次握手:客户端收到SYN+ACK包,再回一个ACK包(ACK=y+1)告诉服务端已经收到
4)三次握手完成,成功建立连接,开始传输数据
四次挥手
image-20220714142224602
1)客户端发送FIN包(FIN=1)给服务端,告诉它自己的数据已经发送完毕,请求终止连接,此时客户端
不发送数据,但还能接收数据
2)服务端收到FIN包,回一个ACK包给客户端告诉它已经收到包了,此时还没有断开socket连接,而是
等待剩下的数据传输完毕
3)服务端等待数据传输完毕后,向客户端发送FIN包,表明可以断开连接
4)客户端收到后,回一个ACK包表明确认收到,等待一段时间,确保服务端不再有数据发过来,然后
彻底断开连接
3.9 说说 TCP 2次握手行不行?为什么要3次
为了实现可靠数据传输, TCP 协议的通信双方, 都必须维护一个序列号, 以标识发送出去的数据
包中, 哪些是已经被对方收到的。 三次握手的过程即是通信双方相互告知序列号起始值, 并确认
对方已经收到了序列号起始值的必经步骤
如果只是两次握手, 至多只有连接发起方的起始序列号能被确认, 另一方选择的序列号则得不到
确认
3.10 简述 TCP 和 UDP 的区别,它们的头部结构是什么样的
参考回答
TCP协议是有连接的,有连接的意思是 开始传输实际数据之前TCP的客户端和服务器端必须通过三
次握手建立连接,会话结束之后也要结束连接。
而UDP是无连接的
TCP协议保证数据按序发送,按序到达,提供超时重传来保证可靠性,但是UDP不保证按序到达,
甚至不保证到达,只是努力交付,即便是按序发送的序列,也不保证按序送到。
TCP协议所需资源多,TCP首部需20个字节(不算可选项),UDP首部字段只需8个字节。
TCP有流量控制和拥塞控制,UDP没有,网络拥堵不会影响发送端的发送速率
TCP是一对一的连接, 而UDP则可以支持一对一,多对多,一对多的通信。
TCP面向的是字节流的服务,UDP面向的是报文的服务。
TCP头部结构如下:
/TCP头定义,共20个字节/
typedef struct _TCP_HEADER
{
// 源端口号16bit
short m_sSourPort;
// 目的端口号16bit
short m_sDestPort;
// 序列号32bit
unsigned int m_uiSequNum;
unsigned int m_uiAcknowledgeNum; // 确认号32bit
// 前4位:TCP头长度;中6位:保留;后6位:标志
short m_sHeaderLenAndFlag;
位
short m_sWindowSize; // 窗口大小16bit
short m_sCheckSum; // 检验和16bit
// 紧急数据偏移量16bit
short m_surgentPointer;
}attribute((packed))TCP_HEADER, PTCP_HEADER;
/
TCP头中的选项定义
kind(8bit)+Length(8bit,整个选项的长度,包含前两部分)+内容(如果有的话)
KIND = 1表示 无操作NOP,无后面的部分
2表示 maximum segment 后面的LENGTH就是maximum segment选项的长度(以byte为单位,
1+1+内容部分长度)
3表示 windows scale 后面的LENGTH就是 windows scale选项的长度(以byte为单位,
1+1+内容部分长度)
4表示 SACK permitted
LENGTH为2,没有内容部分
5表示这是一个SACK包 LENGTH为2,没有内容部分
8表示时间戳,LENGTH为10,含8个字节的时间戳
*/
typedef struct _TCP_OPTIONS
{
char m_ckind;
char m_cLength;
char m_cContext[32];
}attribute((packed))TCP_OPTIONS, *PTCP_OPTIONS;
UDP头部结构如下:
```cpp
/*UDP头定义,共8个字节*/
typedef struct _UDP_HEADER
{
unsigned short m_usSourPort; // 源端口号16bit
unsigned short m_usDestPort; // 目的端口号16bit
unsigned short m_usLength;
unsigned short m_usCheckSum;
// 数据包长度16bit
// 校验和16bit
}__attribute__((packed))UDP_HEADER, *PUDP_HEADER;
3.11 简述 TCP 连接 和 关闭的具体步骤
TCP通过三次握手建立链接:
image-20220714142749715
1)第一次握手:建立连接时,客户端向服务器发送SYN包(seq=x),请求建立连接,等待确认
2)第二次握手:服务端收到客户端的SYN包,回一个ACK包(ACK=x+1)确认收到,同时发送一
个SYN包(seq=y)给客户端
3)第三次握手:客户端收到SYN+ACK包,再回一个ACK包(ACK=y+1)告诉服务端已经收到
4)三次握手完成,成功建立连接,开始传输数据
通过4次挥手关闭链接
image-20220714142827646
1)客户端发送FIN包(FIN=1)给服务端,告诉它自己的数据已经发送完毕,请求终止连接,此
时客户端不发送数据,但还能接收数据
2)服务端收到FIN包,回一个ACK包给客户端告诉它已经收到包了,此时还没有断开socket连
接,而是等待剩下的数据传输完毕
3)服务端等待数据传输完毕后,向客户端发送FIN包,表明可以断开连接
4)客户端收到后,回一个ACK包表明确认收到,等待一段时间,确保服务端不再有数据发过来,
然后彻底断开连接
3.12 简述 TCP 连接 和 关闭的状态转移
参考回答
状态转换如图所示:
image-20220714142914590
上半部分是TCP三路握手过程的状态变迁,下半部分是TCP四次挥手过程的状态变迁。
CLOSED: 起始点,在超时或者连接关闭时候进入此状态,这并不是一个真正的状态,而是这个状
态图的假想起点和终点。
LISTEN: 服务器端等待连接的状态。服务器经过 socket,bind,listen 函数之后进入此状态,开
始监听客户端发过来的连接请求。此称为应用程序被动打开(等到客户端连接请求)。
SYN_SENT: 第一次握手发生阶段,客户端发起连接。客户端调用 connect,发送 SYN 给服务器
端, 然后进入 SYN_SENT 状态,等待服务器端确认(三次握手中的第二个报文)。如果服务器端
不能连接,则直接进入CLOSED状态。
SYN_RCVD:第二次握手 发生阶段,跟 3 对应,这里是服务器端 接收到了 客户端的 SYN, 此时服
务器由 LISTEN 进入 SYN_RCVD状态,同时 服务器端回应一个 ACK,然后再发送一个 SYN 即
SYN+ACK 给客户端。状态图中还描绘了这样一种情况,当客户端在发送 SYN 的同时也收到服务器
端的 SYN请求,即两个同时发起连接请求,那么客户端就会从 SYN_SENT 转换到 SYN_REVD 状
态。
ESTABLISHED:. 第三次握手 发生阶段, 客户端 接收到服务器端的 ACK 包(ACK,SYN)之后,也
会发送一个 ACK 确认包,客户端进入 ESTABLISHED 状态,表明客户端这边已经准备好,但TCP
需要两端都准备好才可以进行数据传输。服务器端收到客户端的 ACK 之后会从 SYN_RCVD 状态转
移到 ESTABLISHED 状态,表明服务器端也准备好进行数据传输了。这样客户端和服务器端都是
ESTABLISHED 状态,就可以进行后面的数据传输了。所以 ESTABLISHED 也可以说是一个数据传
送状态。
下面看看TCP四次挥手过程的状态变迁。
FIN_WAIT_1 :第一次挥手。主动关闭的一方(执行主动关闭的一方既可以是客户端,也可以是服
务器端,这里以客户端执行主动关闭为例),终止连接时,发送 FIN 给对方,然后等待对方返回
ACK 。调用 close() 第一次挥手就进入此状态。
CLOSE_WAIT:**接收到FIN 之后,被动关闭的一方进入此状态**。具体动作是接收到 FIN,同时发送
ACK。之所以叫 CLOSE_WAIT 可以理解为被动关闭的一方此时正在等待上层应用程序发出关闭连
接指令。TCP关闭是全双工过程,这里客户端执行了主动关闭,被动方服务器端接收到FIN 后也需
要调用 close 关闭,这个 CLOSE_WAIT 就是处于这个状态,等待发送 FIN,发送了FIN 则进入
LAST_ACK 状态。
FIN_WAIT_2:主动端(这里是客户端)先执行主动关闭发送FIN,然后接收到 被动方返回的 ACK
后 进入此状态。
LAST_ACK:被动方(服务器端)发起关闭请求,由状态2 进入此状态,具体动作是发送 FIN给对
方,同时在接收到ACK 时进入CLOSED状态。
CLOSING:两边同时发起关闭请求时(即主动方发送FIN,等待被动方返回ACK,同时被动方也发
送了FIN,主动方接收到了FIN之后,发送ACK给被动方),主动方会由FIN_WAIT_1 进入此状态,
等待被动方返回ACK。
TIME_WAIT:从状态变迁图会看到,四次挥手操作最后都会经过这样一个状态然后进入CLOSED
状态。
image-20220714150655902
3.13 简述 TCP 慢启动
参考回答
慢启动(Slow Start),是传输控制协议(TCP)使用的一种阻塞控制机制。慢启动也叫做指数增
长期。慢启动是指每次TCP接收窗口收到确认时都会增长。增加的大小就是已确认段的数目。这种
情况一直保持到要么没有收到一些段,要么窗口大小到达预先定义的阈值。如果发生丢失事件,TCP就认为这是网络阻塞,就会采取措施减轻网络拥挤。一旦发生丢失事件或者到达阈值,TCP就
会进入线性增长阶段。这时,每经过一个RTT窗口增长一个段。
3.14 说说 TCP 如何保证有序
参考回答
主机每次发送数据时,TCP就给每个数据包分配一个序列号并且在一个特定的时间内等待接收主机
对分配的这个序列号进行确认,如果发送主机在一个特定时间内 没有收到接收主机的确认,则发送
主机会重传此数据包。接收主机利用序列号对接收的数据进行确认,以便检测对方发送的数据是否
有丢失或者乱序等,接收主机 一旦收到 已经顺序化的数据,它就将这些数据按正确的顺序重组成数
据流并传递到高层进行处理。
具体步骤如下:
(1)为了保证数据包的可靠传递,发送方 必须把 已发送的数据包保留在缓冲区;
(2)并为 每个已发送的数据包启动 一个超时定时器;
(3)如在定时器超时之前收到了对方发来的应答信息(可能是对本包的应答,也可以是对本包后
续包的应答),则释放该数据包占用的缓冲区;
(4)否则,重传该数据包,直到收到应答或重传次数超过规定的最大次数为止。
(5)接收方收到数据包后,先进行CRC校验,如果正确则把数据交给上层协议,然后给发送方发
送一个累计应答包,表明该数据已收到,如果接收方正好也有数据要发给发送方,应答包也可方在
数据包中捎带过去。
3.15 说说 TCP 常见的拥塞控制算法有哪些
TCP Tahoe/Reno
最初的实现,包括慢启动、拥塞避免两个部分。基于 重传超时(retransmission timeout/RTO)
和 重复确认 为条件 判断是否发生了丢包。
两者的区别在于: Tahoe算法下如果收到三次重复确认,
就进入快重传 立即重发丢失的数据包,同时将慢启动阈值设置为当前拥塞窗口的一半,将拥塞窗口
设置为1MSS,进入慢启动状态 ;而Reno算法如果收到三次重复确认,就进入快重传**,但不进入慢**
启动状态,而是直接将拥塞窗口减半,进入拥塞控制阶段,这称为“快恢复”。
而Tahoe和Reno算法在出现RTO时的措施一致,都是将拥塞窗口降为1个MSS,然后进入慢启动阶
段。
TCP BBR(Bottleneck Bandwidth and Round-trip propagation time)
BBR是由Google设计,于2016年发布的拥塞算法。以往大部分拥塞算法是 基于丢包 来作为降低传
输速率的信号,而BBR则基于模型主动探测。该算法使用 网络最近出站数据分组 当时的最大带宽和
往返时间来建立网络的显式模型。数据包传输的每个累积或选择性确认用于生成记录在数据包传输
过程和确认返回期间的时间内所传送数据量的采样率。
该算法认为 随着网络接口控制器逐渐进入千
兆速度时,分组丢失 不应该被认为 是识别拥塞的主要决定因素 ,所以基于模型的拥塞控制算法能有
更高的吞吐量和更低的延迟,可以用BBR来替代其他流行的拥塞算法,例如CUBIC。
3.16 简述 TCP 超时重传
参考回答:
TCP可靠性 中最重要的一个机制是处理数据超时和重传。TCP协议要求在 发送端每发送一个报文段,就
启动一个定时器并等待确认信息; 接收端成功接收新数据后返回确认信息。若在定时器超时前数据未能
被确认,TCP就认为报文段中的数据已丢失或损坏,需要对报文段中的数据重新组织和重传。
3.17 说说 TCP 可靠性保证
参考回答
TCP主要提供了检验和、序列号/确认应答、超时重传、最大消息长度、滑动窗口控制等方法实现了可靠
性传输。
检验和:
通过检验和的方式,接收端可以检测出来数据是否有差错和异常,假如有差错就会直接丢弃TCP
段,重新发送。TCP在计算检验和时,会在TCP首部加上一个12字节的伪首部。检验和总共计算3
部分:TCP首部、TCP数据、TCP伪首部
image-20220714152005277
序列号/确认应答
序列号/确认应答
这个机制类似于问答的形式。比如在课堂上老师会问你“明白了吗?”,假如你没有隔一段时间没有
回应或者你说不明白,那么老师就会重新讲一遍。其实计算机的确认应答机制也是一样的,发送端
发送信息给接收端,接收端会回应一个包,这个包就是应答包。
image-20220714152041205
上述过程中,只要发送端有一个包传输,接收端没有回应确认包(ACK包),都会重发。或者接收
端的应答包,发送端没有收到也会重发数据。这就可以保证数据的完整性。
超时重传
超时重传
超时重传是指发送出去的数据包到接收到确认包之间的时间,如果超过了这个时间会被认为是丢包
了,需要重传。那么我们该如何确认这个时间值呢?
我们知道,一来一回的时间总是差不多的,都会有一个类似于平均值的概念。比如发送一个包到接
收端收到这个包一共是0.5s,然后接收端回发一个确认包给发送端也要0.5s,这样的两个时间就是
RTT(往返时间)。然后可能由于网络原因的问题,时间会有偏差,称为抖动(方差)。
从上面的介绍来看,超时重传的时间大概是比往返时间+抖动值还要稍大的时间。
image-20220714152153398
但是在重发的过程中,假如一个包经过多次的重发也没有收到对端的确认包,那么就会认为接收端
异常,强制关闭连接。并且通知应用通信异常强行终止。
最大消息长度
在建立TCP连接的时候,双方约定一个最大的长度(MSS)作为发送的单位,重传的时候也是以这
个单位来进行重传。理想的情况下是该长度的数据刚好不被网络层分块。
image-20220714152309375
滑动窗口控制
滑动窗口控制
我们上面提到的超时重传的机制存在效率低下的问题,发送一个包到发送下一个包要经过一段时间
才可以。所以我们就想着能不能不用等待确认包就发送下一个数据包呢?这就提出了一个滑动窗口
的概念。
image-20220714152336817
窗口的大小就是在无需等待确认包的情况下,发送端还能发送的最大数据量。这个机制的实现就是
使用了大量的缓冲区,通过对多个段进行确认应答的功能。通过下一次的确认包可以判断接收端是
否已经接收到了数据,如果已经接收了就从缓冲区里面删除数据。
在窗口之外的数据就是还未发送的和对端已经收到的数据。那么发送端是怎么样判断接收端有没有
接收到数据呢?或者怎么知道需要重发的数据有哪些呢?通过下面这个图就知道了。
image-20220714152449296
如上图,接收端在没有收到自己所期望的序列号数据之前,会对之前的数据进行重复确认。发送端
在收到某个应答包之后,又连续3次收到同样的应答包,则数据已经丢失了,需要重发。
拥塞控制:
窗口控制解决了 两台主机之间因传送速率而可能引起的丢包问题,在一方面保证了TCP数据传送
的可靠性。然而如果网络非常拥堵,此时再发送数据就会加重网络负担,那么发送的数据段很可能
超过了最大生存时间也没有到达接收方,就会产生丢包问题。为此TCP引入慢启动机制,先发出少
量数据,就像探路一样,先摸清当前的网络拥堵状态后,再决定按照多大的速度传送数据。
发送开始时定义拥塞窗口大小为1;每次收到一个ACK应答,拥塞窗口加1;而在每次发送数据时,
发送窗口取拥塞窗口与接送段接收窗口最小者。
慢启动:在启动初期以指数增长方式增长;设置一个慢启动的阈值,当以指数增长达到阈值时就停
止指数增长,按照线性增长方式增加至拥塞窗口;线性增长达到网络拥塞时立即把拥塞窗口置回
1,进行新一轮的“慢启动”,同时新一轮的阈值变为原来的一半。
image-20220714152648524
3.18 简述 TCP 滑动窗口以及重传机制
参考回答
滑动窗口协议是传输层进 行流控的一种措施,接收方 通过 通告发送方 自己的窗口大小,从而控制发
送方的发送速度,从而达到防止发送方发送速度过快而导致自己被淹没的目的。TCP的滑动窗口解决了端到端的流量控制问题,允许接受方对传输进行限制,直到它拥有足够的缓冲空间来容纳更多的数据。
TCP在发送数据时会设置一个计时器,若到计时器超时仍未收到数据确认信息,则会引发相应的超
时或基于计时器的重传操作,计时器超时称为重传超时(RTO) 。另一种方式的重传称为 快速重
传,通常发生在没 有延时的情况下。若TCP累积确认无法返回新的ACK,或者当ACK包含的选择确
认信息(SACK)表明出现失序报文时,快速重传会推断出现丢包,需要重传。
3.19 说说滑动窗口过小怎么办
参考回答
我们可以假设窗口的大小是1,也是就每次只能发送一个数据,并且发送方只有接受方对这个数据
进行确认了以后才能发送下一个数据。如果说窗口过小,那么当传输比较大的数据的时候需要不 停
的对数据进行确认,这个时候就会造成很大的延迟。
3.20 说说如果三次握手时候每次握手信息对方没收到会怎么样,分情况介绍
参考回答
如果第一次握手消息丢失,那么请求方不会得到ack消息,超时后进行重传
如果第二次握手消息丢失,那么请求方不会得到ack消息,超时后进行重传
如果第三次握手消息丢失,那么Server 端该TCP连接的状态为SYN_RECV,并且会根据 TCP的超时重
传机制,会等待3秒、6秒、12秒后重新发送SYN+ACK包,以便Client重新发送ACK包。而Server
重发SYN+ACK包的次数,可以设置/proc/sys/net/ipv4/tcp_synack_retries修改, 默认值为5.如果
重发指定次数之后,仍然未收到 client 的ACK应答,那么一段时间后,Server自动关闭这个连接。
client 一般是通过 connect() 函数来连接服务器的,而connect()是在 TCP的三次握手的第二次握手
完成后就成功返回值。也就是说 client 在接收到 SYN+ACK包,它的TCP连接状态就为 established
(已连接),表示该连接已经建立。那么如果 第三次握手中的ACK包丢失的情况下,Client 向
server端发送数据,Server端将以 RST包响应,方能感知到Server的错误。
3.21 简述 TCP 的 TIME_WAIT,为什么需要有这个状态
参考回答
TIME_WAIT状态 也成为2MSL等待状态。每个具体TCP实现必须选择一个报文段最大生存时间
MSL(Maximum Segment Lifetime),它是任何报文段被丢弃前在网络内的最长时间。这个时
间是有限的, 因为TCP报文段以IP数据报在网络内传输,而IP数据报则 有限制其生存时间的TTL字
段。
对一个具体实现所给定的MSL值,处理的原则是 :当TCP执行一个主动关闭,并发回最后一个
ACK,该连接必须在TIME_WAIT状态停留的时间为2倍的MSL。这样可让TCP再次发送最后的ACK
以防这个ACK丢失(另一端超时并重发最后的FIN)。
这种2MSL等待的另一个结果是这个TCP连接在2MSL等待期间,定义这个连接的插口(客户的IP地
址和端口号,服务器的IP地址和端口号)不能再被使用。这个连接只能在2MSL结束后才能再被使
用。
理论上,四个报文都发送完毕,就可以直接进入CLOSE状态了,但是可能网络是不可靠的,有可
能最后一个ACK丢失。所以TIME_WAIT状态 就是用来重发可能丢失的ACK报文。
3.22 简述什么是 MSL,为什么客户端连接要等待2MSL的时间才能完全关闭
参考回答
MSL是Maximum Segment Lifetime的英文缩写,可译为“最长报文段寿命”,它是任何报文在网络
上存在的最长时间,超过这个时间报文将被丢弃。
为了保证客户端发送的最后一个ACK报文段能够到达服务器。因为这个ACK有可能丢失,从而导致
处在LAST-ACK状态的服务器收不到对FIN-ACK的确认报文。服务器会超时重传这个FIN-ACK,接着
客户端再重传一次确认,重新启动时间等待计时器。最后客户端和服务器都能正常的关闭。假设客
户端不等待2MSL,而是在发送完ACK之后直接释放关闭,一但这个ACK丢失的话,服务器就无法
正常的进入关闭连接状态。
两个理由:
1.保证客户端发送的最后一个ACK报文段能够到达服务端。
这个ACK报文段有可能丢失,使得处于LAST-ACK状态的B 收不到 对已发送的FIN+ACK报文段
的确认,服务端超时重传FIN+ACK报文段,而客户端能在2MSL时间内收到这个重传的
FIN+ACK报文段,接着客户端重传一次确认,重新启动2MSL计时器,最后客户端和服务端都
进入到CLOSED状态, 若客户端在TIME-WAIT状态不等待一段时间,而是发送完ACK报文段后
立即释放连接,则无法收到服务端重传的FIN+ACK报文段,所以 不会再发送一次确认报文
段,则服务端无法正常进入到CLOSED状态
防止“已失效的连接请求报文段”出现在本连接中。
客户端在发送完最后一个ACK报文段后,再经过2MSL,就可以使本连接持续的时间内所产生
的所有报文段都从网络中消失,使下一个新的连接中不会出现这种旧的连接请求报文段。
3.23 说说什么是 SYN flood,如何防止这类攻击?
参考回答
SYN Flood是当前最流行的DoS(拒绝服务攻击)与DDoS(分布式拒绝服务攻击)的方式之一,这是
一种利用TCP协议缺陷,发送大量伪造的TCP连接请求,使被攻击方资源耗尽(CPU满负荷或内存
不足)的攻击方式.
有以下三种方法预防或响应网络上的DDoS攻击:
(1)从互联网服务提供商(ISP)购买服务。
许多互联网服务提供商(ISP)提供DDoS缓解服务,但是当企业网络受到攻击时,企业 需要向互联网
服务提供商(ISP)报告事件以开始缓解。**这种策略称为“清洁管道”,在互联网服务提供商(ISP)收取服
务费用时很受欢迎,但在缓解措施开始之前,通常会导致30到60分钟的网络延迟**。
(2)保留在内部并自己解决。
企业可以使用入侵防御系统/防火墙技术和专用于防御DDoS攻击的专用硬件来实现内部预防和响应
DDoS攻击。不幸的是,受影响的流量已经在网络上消耗了宝贵的带宽。这使得该方法最适合在托
管设施中配备设备的企业,在这些企业中,流量是通过交叉连接到达互联网服务提供商(ISP),从
而保护流向企业其他部门的下游带宽。
(3)使用内容分发网络(CDN)。
由于IT团队可以将基础设施置于内容分发网络(CDN)后面,因此这种方法可以最大程度地减少对企
业网络基础设施的攻击。这些网络庞大而多样,如果组织订阅DNS和DDoS缓解措施,则它们可以
保护电子商务站点以及企业本身。
3.24 说说什么是 TCP 粘包和拆包?
参考回答
TCP是个“流”协议,所谓流,就是没有界限的一串数据。大家可以想想河里的流水,是连成一片
的,其间并没有分界线。TCP底层并不了解上层业务数据的具体含义,它会根据TCP缓冲区的实际
情况进行包的划分,所以在业务上认为,一个完整的包可能会被TCP拆分成多个包进行发送,也有
可能把多个小的包封装成一个大的数据包发送,这就是所谓的TCP粘包和拆包问题。
假设客户端分别发送了两个数据包D1和D2给服务端,由于服务端一次读取到的字节数是不确定的,故
可能存在以下4种情况。
(1)服务端 分两次读取到了两个独立的数据包,分别是D1和D2,没有粘包和拆包;
(2)服务端 一次接收到了两个数据包,D1和D2 粘合在一起,被称为TCP粘包;
(3)服务端分两次读取到了两个数据包,第一次读取到 了完整的D1包和D2包的部分内容, 第二次读取
到了D2包的剩余内容,这被称为TCP拆包;
(4)服务端分两次读取到了两个数据包,第一次读取到了D1包的部分内容D1_1,第二次读取到了D1
包的剩余内容D1_2和D2包的整包。
如果此时服务端TCP接收滑窗非常小,而数据包D1和D2比较大,很有可能会发生第五种可能,即服务
端分多次才能将D1和D2包接收完全,期间发生多次拆包。
3.25 说说 TCP 与 UDP 在网络协议中的哪一层,他们之间有什么区别?
TCP和UDP协议都是传输层协议。二者的区别主要有:
基于连接vs无连接
TCP是面向连接的协议。
UDP是无连接的协议。UDP更加适合消息的多播发布,从单个点向多个点传输消息。
可靠性
TCP提供交付保证,传输过程中丢失,将会重发。
UDP是不可靠的,不提供任何交付保证。(网游和视频的丢包情况)
有序性
TCP保证了消息的有序性,即使到达客户端顺序不同,TCP也会排序。
UDP不提供有序性保证。
数据边界
TCP不保存数据边界。
虽然TCP也将在收集所有字节之后生成一个完整的消息,但是这些信息在传给传输给接受端
之前将储存在TCP缓冲区,以确保更好的使用网络带宽。
UDP保证。在UDP中,数据包单独发送的,只有当他们到达时,才会再次集成。包有明确的界限来哪些
包已经收到,这意味着在消息发送后,在接收器接口将会有一个读操作,来生成一个完整的
消息。
速度
TCP速度慢
UDP速度快。应用在在线视频媒体,电视广播和多人在线游戏。
发送消耗
TCP是重量级。
UDP是轻量级。
因为UDP传输的信息中不承担任何间接创造连接,保证交货或秩序的的信息。
这也反映在用于报头大小。
报头大小
TCP头大。
一个TCP数据包报头的大小是20字节。
TCP报头中包含序列号,ACK号,数据偏移量,保留,控制位,窗口,紧急指针,可选项,填
充项,校验位,源端口和目的端口。
UDP头小。
UDP数据报报头是8个字节。
而UDP报头只包含长度,源端口号,目的端口,和校验和。
拥塞或流控制
TCP有流量控制。
在任何用户数据可以被发送之前,TCP需要三数据包来设置一个套接字连接。TCP处理的可靠
性和拥塞控制。
UDP不能进行流量控制。
应用
由于TCP提供可靠交付和有序性的保证,它是最适合需要高可靠并且对传输时间要求不高的应用。
UDP是更适合的应用程序需要快速,高效的传输的应用,如游戏。
UDP是无状态的性质,在服务器端需要对大量客户端产生的少量请求进行应答的应用中是非常有
用的。
在实践中,TCP被用于金融领域,如FIX协议是一种基于TCP的协议,而UDP是大量使用在游戏和娱
乐场所。
10.上层使用的协议
基于TCP协议的:Telnet,FTP以及SMTP协议。
基于UDP协议的:DHCP、DNS、SNMP、TFTP、BOOTP。
3.26 说说从系统层面上,UDP 如何保证尽量可靠?
参考回答
UDP仅提供了最基本的数据传输功能,至于传输时连接的建立和断开、传输可靠性的保证这些
UDP统统不关心,而是把这些问题抛给了UDP上层的应用层程序去处理,自己仅提供传输层协议
的最基本功能。
最简单的方式 是在 应用层模仿传输层TCP的可靠性传输。下面不考虑拥塞处理,可靠UDP的简单设
计。
添加seq/ack机制,确保数据发送到对端
添加发送和接收缓冲区,主要是用户超时重传。
添加超时重传机制。
3.27 说一说 TCP 的 keepalive,以及和 HTTP 的 keepalive 的区别?
HTTP Keep-Alive
在http早期,每个http请求都要求打开一个tpc socket连接,并且使用一次之后就断开这个tcp连
接。使用keep-alive可以改善这种状态,即在一次TCP连接中可以持续发送多份数据而不会断开连
接。通过使用keep-alive机制,可以减少tcp连接建立次数,也意味着可以减少TIME_WAIT状态连
接,以此提高性能和提高httpd服务器的吞吐率(更少的tcp连接意味着更少的系统内核调用,socket
的accept()和close()调用)。 但是,keep-alive并不是免费的午餐,长时间的tcp连接容易导致系统资
源无效占用。配置不当的keep-alive,有时比重复利用连接带来的损失还更大。所以,正确地设置
keep-alive timeout时间非常重要。
TCP KEEPALIVE
链接建立之后,如果应用程序或者上层协议一直不发送数据,或者隔很长时间才发送一次数据,当
链接很久没有数据报文传输时 去确定对方还在线,到底是掉线了还是确实没有数据传输,链接
还需不需要保持,这种情况在TCP协议设计中是需要考虑到的。TCP协议通过一种巧妙的方式去解
决这个问题,当超过一段时间之后,TCP自动发送一个数据为空的报文给对方,如果对方回应了这
个报文,说明对方还在线,链接可以继续保持,如果对方没有报文返回,并且重试了多次之后则认
为链接丢失,没有必要保持链接。
TCP的keepalive机制和HTTP的keep-alive机制是说的完全不同的两个东西, tcp的keepalive是在
ESTABLISH状态的时候,双方如何检测连接的可用行。而http的keep-alive说的 是如何避免进行重
复的TCP三次握手和四次挥手的环节。
3.28 简述 TCP 协议的延迟 ACK 和累计应答
参考回答
延迟应答指的 是:TCP在接收到对端的报文后,并不会立即发送ack,而是等待一段时间发送ack,
以便将ack和要发送的数据一块发送。当然ack不能无限延长,否则对端会认为包超时而造成报文
重传。linux采用 动态调节算法来确定延时的时间。
累计应答指的是:为了保证顺序性,每一个包都有一个ID(序号),在建立连接的时候,会商定
起始的ID是多少,然后按照ID一个个发送。而为了保证不丢包,对应发送的包都要进行应答,但
不是一个个应答,而是会应答某个之前的ID,该模式称为累计应答
3.29 说说 TCP 如何加速一个大文件的传输
参考回答
建连优化:TCP 在建立连接时,如果丢包,会进入重试,重试时间是 1s、2s、4s、8s 的指数递增
间隔,缩短定时器可以让 TCP 在丢包环境建连时间更快,非常适用于高并发短连接的业务场景。
平滑发包: 在 RTT 内均匀发包,规避 微分时间内的流量突发,尽量避免瞬间拥塞
丢包预判: 有些网络的丢包是有规律性的,例如每隔一段时间出现一次丢包,例如每次丢包都连续
丢几个等,如果程序能自动发现这个规律(有些不明显),就可以针对性提前多发数据,减少重传
时间、提高有效发包率。
RTO 探测: 若始终收不到 ACK 报文,则需要触发 RTO 定时器。RTO 定时器一般都时间非常长,
会浪费很多等待时间,而且一旦 RTO,CWND 就会骤降(标准 TCP),因此利用 Probe 提前与
RTO 去试探,可以规避由于 ACK 报文丢失而导致的速度下降问题。
带宽评估: 通过单位时间内收到的 ACK 或 SACK 信息可以得知客户端有效接收速率,通过这个速
率可以更合理的控制发包速度。
带宽争抢: 有些场景(例如合租)是大家互相挤占带宽的,假如你和室友各 1Mbps 的速度看电
影,会把 2Mbps 出口占满,而如果一共有 3 个人看,则每人只能分到 1/3。若此时你的流量流量
达到 2Mbps,而他俩还都是 1Mbps,则你至少仍可以分到 2/(2+1+1) * 2Mbps = 1Mbps 的 50%
的带宽,甚至更多,代价就是服务器侧的出口流量加大,增加成本。(TCP 优化的本质就是用带
宽换用户体验感)
3.30 服务器怎么判断客户端断开了连接
参考回答
检测连接是否丢失的方法大致有两种:keepalive和heart-beat
(tcp内部机制) 采用keepalive,它会先要求此连接一定时间没有活动(一般是几个小时),然后
发出数据段,经过多次尝试后(每次尝试之间也有时间间隔),如果仍没有响应,则判断连接中
断。可想而知,整个周期需要很长的时间。
(应用层实现)一个简单的heart-beat实现 一般测试连接是否 中断采用的时间间隔都比较短,可以
很快的决定连接是否中断。并且,由于是在应用层实现,因为可以自行决定当判断连接中断后应该
采取的行为,而keepalive在判断连接失败后 只会将连接丢弃。
3.31 说说端到端,点到点的区别
参考回答
端到端通信 是针对传输层来说的, 传输层 为网络中的主机提供 端到端的通信。因为无论tcp还是
udp协议,都要负责把 上层交付的数据 从发送端传输到接收端,不论其中间跨越多少节点。只不过
tcp比较可靠而udp不可靠而已。所以称之为端到端,也就是从发送端 到 接收端。
它是一个网络连接,指的是在数据传输之前,在发送端 与 接收端之间(忽略中间有多少设备)为数
据的传输建立一条链路,链路建立以后,发送端就可以发送数据,知道数据发送完毕,接收端确认
接收成功。 也就是说在数据传输之前,先为数据的传输开辟一条通道,然后在进行传输。从发送
端发出数据到接收端接收完毕,结束。
端到端通信 建立在 点到点通信的基础之上,它是由一段段的 点到点通信信道构成的,是比 点到点通
信 更高一级的通信方式,完成应用程序(进程)之间的通信。
端到端的优点:
链路建立之后,发送端 知道 接收端一定能收到,而且经过中间交换设备时不需要进行存储转发,因
此传输延迟小。
端到端传输的缺点:
(1)直到 接收端 收到数据为止,发送端的设备一直要参与传输。如果整个传输的延迟很长,那么
对发送端的设备造成很大的浪费。
(2)如果接收设备 关机或故障,那么端到端传输不可能实现。
2.
点到点通信是针对 数据链路层或网络层来说的,因为 数据链路层 只负责 直接相连的两个节点之间的
通信,一个节点的数据链路层接受 ip层数据 并封装之后,就把数据帧 从链路上发送到与其相邻的下
一个节点。 点对点是基于MAC地址和或者IP地址,是指一 个设备 发数据给与该这边直接连接的其
他设备,这台设备又在合适的时候 将数据传递给与它相连的下一个设备,通过一台一台直接相连的
设备把数据传递到接收端。
直接相连的节点对等实体的通信叫点到点通信。它只提供一台机器到另一台机器之间的通信,不会
涉及到程序或进程的概念。同时 点到点通信并不能保证数据传输的可靠性,也不能说明源主机与目
的主机之间是哪两个进程在通信。
由物理层、数据链路层和网络层组成的通信子网为网络环境中的主机提供点到点的服务
点到点的优点:
(1)发送端设备 送出数据后,它的任务已经完成,**不需要参与整个传输过程,这样不会浪费发送**
端设备的资源。
(2)即使接收端设备关机或故障,点到点传输也可以采用存储转发技术进行缓冲。
点到点的缺点:
点到点传输的缺点是 发送端发出数据后,不知道 接收端 能否收到或何时能收到数据。
在一个网络系统的不同分层中,可能用到端到端传输,也可能用到点到点传输。如Internet网,IP
及以下各层采用点到点传输,4层以上采用端到端传输。
3.32 说说浏览器从输入 URL 到展现页面的全过程
参考回答
1、输入地址
2、浏览器查找域名的 IP 地址
3、浏览器向 web 服务器发送一个 HTTP 请求
4、服务器的永久重定向响应
6、服务器处理请求
7、服务器返回一个 HTTP 响应
8、浏览器显示 HTML
9、浏览器发送请求获取嵌入在 HTML 中的资源(如图片、音频、视频、CSS、JS等等)
3.33 简述 HTTP 和 HTTPS 的区别?
参考回答
HTTP:是互联网上应用最为广泛的一种网络协议,是一个客户端和服务器端请求和应答的标准
(TCP),用于从WWW服务器传输超文本到本地浏览器的传输协议,它可以使浏览器更加高效,
使网络传输减少。
HTTPS:是以安全为目标的HTTP通道,简单讲是HTTP的安全版,即HTTP下加入SSL层,HTTPS
的安全基础是SSL,因此加的详细内容就需要SSL。
HTTPS协议的主要作用可
以分为两种:一种是建立一个信息安全通道,来保证数据传输的安全;
另一种就是确认网站的真实性。
HTTP与HTTPS的区别
https协议需要到ca申请证书,一般免费证书较少,因而需要一定费用。
http是超文本传输协议,信息是明文传输,https则是具有安全性的ssl加密传输协议。
http和https使用的是完全不同的连接方式,用的端口也不一样,前者是80,后者是443。
http的连接很简单,是无状态的;HTTPS协议是由SSL+HTTP协议构建的可进行加密传输、身份认
证的网络协议,比http协议安全。
3.34 说说 HTTP 中的 referer 头的作用
参考回答:
HTTP Referer是header的一部分,当浏览器向web服务器发送请求的时候,一般会带上Referer,
告诉服务器该网页 是从 哪个页面链接过来的,服务器因此可以获得一些信息用于处理。
防盗链。假如在www.google里有一个www.baidu链接,那么点击进入这个www.ba
idu,它的header信息里就有:Referer= http://www.google
只允许我本身的网站访问本身的图片服务器,假如域是 www.google ,那么图片服务器每次
取到Referer来判断一下域名是不是 www.google , 如果是就继续访问,不是就拦截。
将这个http请求发给服务器后,如果服务器要求必须是某个地址或者某几个地址才能访问,而你发
送的referer不符合他的要求,就会拦截或者跳转到他要求的地址,然后再通过这个地址进行访
问。
防止恶意请求
比如 静态请求是 *.html 结尾的,动态请求是 *.shtml ,那么由此可以这么用,所有的 *.shtml
请求,必须Referer为我自己的网站。
空Referer
定义: Referer头部的内容为空,或者,一个HTTP请求中根本不包含Referer头部(一个请求并不
是由链接触发产生的)
直接在浏览器的地址栏中输入一个资源的URL地址,那么这种请求是不会包含Referer字段的,因
为这是一个“凭空产生”的HTTP请求,并不是从一个地方链接过去的。
那么在防盗链设置中,允许空Referer和不允许空Referer有什么区别?
允许Referer为空,意味着你允许比如浏览器直接访问。
防御CSRF
比对HTTP 请求的来源地址,如果Referer中的地址是安全可信任的地址,那么就放行
3.35 说说 HTTP 的方法有哪些
参考回答
GET: 用于请求访问已经被URI(统一资源标识符)识别的资源,可以通过URL传参给服务器
POST:用于 传输信息给 服务器,主要功能与GET方法类似,但一般推荐使用POST方式。
PUT: 传输文件,报文主体中 包含 文件内容,保存到对应URI位置。
HEAD: 获得报文首部,与GET方法类似,只是不返回报文主体,一般用于验证URI是否有效。
DELETE:删除文件,与PUT方法相反,删除对应URI位置的文件。
OPTIONS:查询相应URI支持的HTTP方法。
3.36 简述 HTTP 1.0,1.1,2.0 的主要区别
参考回答:
http/1.0 :
默认不支持长连接,需要设置keep-alive参数指定
强缓存expired、协商缓存last-modified\if-modified-since 有一定的缺陷
http 1.1 :
1.默认长连接(keep-alive),http请求可以复用Tcp连接,但是同一时间只能对应一个http请求(http请
求在一个Tcp中是串行的)
2.增加了强缓存cache-control、协商缓存etag\if-none-match 是对http/1 缓存的优化
http/2.0 :
1.多路复用,一个Tcp中多个http请求是并行的 (雪碧图、多域名散列等优化手段http/2中将变得多
余)
2.二进制格式编码传输
3.使用HPACK算法做header压缩
4.服务端推送
3.37 说说 HTTP 常见的响应状态码及其含义
参考回答:
参考回答
200 : 从状态码发出的请求被服务器正常处理。
204 : 服务器接收的请求已成功处理,但在返回的响应报文中不含实体的主体部分【即没有内
容】。
206 : 部分的内容(如:客户端进行了范围请求,但是服务器成功执行了这部分的干请求)。
301 : 跳转,代表永久性重定向(请求的资源已被分配了新的URI,以后已使用资源,现在设置了
URI)。
302 : 临时性重定向(请求的资源已经分配了新的URI,希望用户本次能够使用新的URI来进行访
问)。
303 : 由于请求对应的资源存在的另一个URI(因使用get方法,定向获取请求的资源)。
304 : 客户端发送附带条件的请求时,服务器端允许请求访问资源,但因发生请求未满足条件的情
况后,直接返回了 304。
307 : 临时重定向【该状态码与302有着相同的含义】。
400 : 请求报文中存在语法错误(当错误方式时,需修改请求的内容后,再次发送请求)。
401 : 发送的请求需要有通过HTTP认证的认证信息。
403 : 对请求资源的访问被服务器拒绝了。
404 : 服务器上无法找到请求的资源。
500 : 服务器端在执行请求时发生了错误。
503 : 服务器暂时处于超负载或者是正在进行停机维护,现在无法处理请求。
答案解析
1XX : 信息类状态码(表示接收请求状态处理)
2XX : 成功状态码(表示请求正常处理完毕)
3XX : 重定向(表示需要进行附加操作,已完成请求)
4XX : 客户端错误(表示服务器无法处理请求)
5XX : 服务器错误状态码(表示服务器处理请求的时候出错)
3.38 说说 GET请求和 POST 请求的区别
参考回答:
GET请求在URL中传送的参数是有长度限制的,而POST没有。
GET比POST更不安全,因为参数直接暴露在URL上,所以不能用来传递敏感信息。
GET参数 通过URL传递,POST放在Request body中。
GET请求参数 会被完整保留在浏览器历史记录里,而POST中的参数不会被保留。
GET请求只能进行url编码,而POST支持多种编码方式。
GET请求会被浏览器主动cache,而POST不会,除非手动设置。
GET产生的URL地址可以被Bookmark,而POST不可以。
GET在浏览器回退时是无害的,而POST会再次提交请求。
3.39 说说 Cookie 和 Session 的关系和区别是什么
参考回答:
Cookie与Session都是 会话的一种方式。它们的典型使用场景比如“购物车”,当你点击下单按钮
时,服务端 并不清楚具体用户的具体操作,为了标识并跟踪该用户,了解购物车中有几样物品,服
务端通过为该用户创建Cookie/Session来获取这些信息。
cookie数据 存放在客户的浏览器上,session数据放在服务器上。
cookie不是很安全,别人可以分析存放在本地的COOKIE并进行COOKIE欺骗 考虑到安全应当使
用session。
session会在一定时间内保存在服务器上。当访问增多,会比较占用你服务器的性能 考虑到减轻
服务器性能方面,应当使用COOKIE。
单个cookie保存的数据不能超过4K,很多浏览器都限制一个站点最多保存20个cookie。
3.40 简述 HTTPS 的加密与认证过程
参考回答:
客户端 在 浏览器中 输入一个https网址,然后连接到server的443端口 采用https协议的server必须
有一套数字证书(一套公钥和密钥) 首先server 将证书(公钥)传送到客户端 客户端解析证书,
验证成功,则生成一个随机数(私钥),并用 证书 将 该随机数加密后 传回server server 用密钥解
密后,获得这个随机值,然后将要传输的信息和私钥通过某种算法混合在一起(加密)传到客户端
客户端 用之前的生成的随机数(私钥)解密服务器端传来的信息
首先 浏览器 会从 内置的证书列表中索引,找到服务器下发证书对应的机构**,如果没有找到,此时就
会提示用户该证书是不是由权威机构颁发,是不可信任的。如果查到了对应的机构,则取出该机构
颁发的公钥**。
用 机构的证书公钥 解密得到证书的内容和证书签名,内容 包括网站的网址、网站的公钥、证书的有
效期等。浏览器 会先验证证书签名的合法性。签名通过后,浏览器验证证书记录的网址是否和当前
网址是一致的,不一致会提示用户。如果网址一致会检查证书有效期,证书过期了也会提示用户。
这些都通过认证时,浏览器就可以安全使用证书中的网站公钥了。
4. 数据库
4.1 SQL
4.1.1 介绍一下数据库分页
参考答案
MySQL的分页语法:
在MySQL中,SELECT语句默认返回所有匹配的行,它们可能是指定表中的每个行。为了返回第一行或
前几行,可使用LIMIT子句,以实现分页查询。LIMIT子句的语法如下:
-- 在所有的查询结果中,返回前5行记录。
SELECT prod_name FROM products LIMIT 5;
-- 在所有的查询结果中,从第5行开始,返回5行记录。
SELECT prod_name FROM products LIMIT 5,5;
总之,带一个值的LIMIT总是从第一行开始,给出的数为返回的行数。带两个值的LIMIT可以指定从行号
为第一个值的位置开始。
优化LIMIT分页:
在偏移量非常大的时候,例如 LIMIT 10000,20 这样的查询,这时MySQL需要查询10020条记录然后
只返回最后20条,前面的10000条记录都将被抛弃,这样的代价是非常高的。如果所有的页面被访问的
频率都相同,那么这样的查询平均需要访问半个表的数据。要优化这种查询,要么是在页面中限制分页
的数量,要么是优化大偏移量的性能。
优化此类分页查询的一个最简单的办法就是尽可能地使用索引覆盖扫描,而不是查询所有的列,然后根
据需要做一次关联操作再返回所需的列。对于偏移量很大的时候,这样做的效率会提升非常大。考虑下
面的查询:
SELECT film_id,description FROM sakila.film ORDER BY title LIMIT 50,5;
如果这个表非常大,那么这个查询最好改写成下面的样子:
SELECT film.film_id,film.description
FROM sakila.film
INNER JOIN (
SELECT film_id FROM sakila.film ORDER BY title LIMIT 50,5
) AS lim USING(film_id);
这里的“延迟关联”将大大提升查询效率,它让MySQL扫描尽可能少的页面,获取需要访问的记录后再根
据关联列回原表查询需要的所有列。这个技术也可以用于优化关联查询中的LIMIT子句。
有时候也可以将LIMIT查询转换 为已知位置的查询,让MySQL通过范围扫描获得对应的结果。例如,如
果在一个位置列上有索引,并且预先计算出了边界值,上面的查询就可以改写为:
SELECT film_id,description FROM skila.film
WHERE position BETWEEN 50 AND 54 ORDER BY position;
对数据进行排名的问题也与此类似,但往往还会同时和GROUP BY混合使用,在这种情况下通常都需要
预先计算并存储排名信息。
LIMIT和OFFSET的问题,其实是OFFSET的问题,它会导致MySQL扫描大量不需要的行然后再抛弃掉。
如果可以使用书签记录上次取数的位置,那么下次就可以直接从该书签记录的位置开始扫描,这样就可
以避免使用OFFSET。例如,若需要按照租赁记录做翻页,那么可以根据最新一条租赁记录向后追溯,
这种做法可行是因为租赁记录的主键是单调增长的。首先使用下面的查询获得第一组结果:
SELECT * FROM sakila.rental ORDER BY rental_id DESC LIMIT 20;
假设上面的查询返回的是主键16049到16030的租赁记录,那么下一页查询就可以从16030这个点开
始:
SELECT * FROM sakila.rental
WHERE rental_id < 16030 ORDER BY rental_id DESC LIMIT 20;
4.1.2 介绍一下SQL中的聚合函数
参考答案
常用的聚合函数有COUNT()、AVG()、SUM()、MAX()、MIN(),下面以MySQL为例,说明这些函数的作
用。
COUNT()函数:
COUNT()函数统计数据表中包含的记录行的总数,或者根据查询结果返回列中包含的数据行数,它有两
种用法:
COUNT(*) 计算表中总的行数,不管某列是否有数值或者为空值。
COUNT(字段名) 计算指定列下总的行数,计算时将忽略空值的行。
COUNT()函数可以与GROUP BY一起使用来计算每个分组的总和。
AVG()函数():
AVG()函数通过计算返回的行数和每一行数据的和,求得指定列数据的平均值。
AVG()函数可以与GROUP BY一起使用,来计算每个分组的平均值。
SUM()函数:
SUM()是一个求总和的函数,返回指定列值的总和。
SUM()可以与GROUP BY一起使用,来计算每个分组的总和。
MAX()函数:
MAX()返回指定列中的最大值。
MAX()也可以和GROUP BY关键字一起使用,求每个分组中的最大值。
MAX()函数不仅适用于查找数值类型,也可应用于字符类型。
MIN()函数:
MIN()返回查询列中的最小值。
MIN()也可以和GROUP BY关键字一起使用,求出每个分组中的最小值。
MIN()函数与MAX()函数类似,不仅适用于查找数值类型,也可应用于字符类型。
4.1.3 表跟表是怎么关联的?
参考答案
表与表之间常用的关联方式有两种:内连接、外连接,下面以MySQL为例来说明这两种连接方式。
内连接:
内连接通过INNER JOIN来实现,它将返回两张表中满足连接条件的数据,不满足条件的数据不会查询出
来。
外连接:外连接通过OUTER JOIN来实现,它会返回两张表中满足连接条件的数据,同时返回不满足连接条件的
数据。外连接有两种形式:左外连接(LEFT OUTER JOIN)、右外连接(RIGHT OUTER JOIN)。
左外连接: 可以简称为左连接(LEFT JOIN),它会返回左表中的所有记录和右表中满足连接条件
的记录。
右外连接:可以简称为右连接(RIGHT JOIN),它会返回右表中的所有记录和左表中满足连接条
件的记录。
除此之外,还有一种常见的连接方式:等值连接。这种连接 是通过WHERE子句中的条件,将两张表连
接在一起,它的实际效果等同于内连接。出于语义清晰的考虑,一般更建议使用内连接,而不是等值连
接。
以上是从语法上来说明表与表之间关联的实现方式,而从表的关系上来说,比较常见的关联关系有:一
对多关联、多对多关联、自关联。
一对多关联:这种关联形式最为常见,一般是两张表具有主从关系,并且以主表的主键关联从表的
外键来实现这种关联关系。另外,以从表的角度来看,它们是具有多对一关系的,所以不再赘述多
对一关联了。
多对多关联:这种关联关系比较复杂,如果两张表具有多对多的关系,那么它们之间需要有一张中
间表来作为衔接,以实现这种关联关系。这个中间表要设计两列,分别存储那两张表的主键。因
此,这两张表中的任何一方,都与中间表形成了一对多关系,从而在这个中间表上建立起了多对多
关系。
自关联:自关联就是一张表自己与自己相关联,为了避免表名的冲突,需要在关联时通过别名将它
们当做两张表来看待。一般在表中数据具有层级(树状)时,可以采用自关联一次性查询出多层级
的数据
4.1.4 说一说你对外连接的了解
外连接通过 OUTER JOIN来实现,它会返回两张表中满 足连接条件的数据,同时返回不满足连接条件的
数据。常见的外连接有两种形式:左外连接(LEFT OUTER JOIN)、右外连接(RIGHT OUTER
JOIN)。
左外连接:可以简称为左连接(LEFT JOIN),它会返回左表中的所有记录和右表中满足连接条件
的记录。
右外连接:可以简称为右连接(RIGHT JOIN),它会返回右表中的所有记录和左表中满足连接条
件的记录。
实际上,外连接还有一种形式:完全外连接(FULL OUTER JOIN),但MySQL不支持这种形式。
4.1.6 SQL中怎么将行转成列?
我们以MySQL数据库为例,来说明行转列的实现方式。
首先,假设我们有一张分数表(tb_score),表中的数据如下图:
image-20220714220424145
然后,我们再来看一下转换之后需要得到的结果,如下图:
image-20220714220459092
、
可以看出,这里 行转列是将原来的 subject字段的多行内容选出来,作为结果集中的不同列,并根据
userid进行分组显示对应的score。通常,我们有两种方式来实现这种转换。
使用 CASE...WHEN...THEN 语句实现行转列,参考如下代码:
SELECT userid,
SUM(CASE `subject` WHEN '语文' THEN score ELSE 0 END) as '语文',
SUM(CASE `subject` WHEN '数学' THEN score ELSE 0 END) as '数学',
SUM(CASE `subject` WHEN '英语' THEN score ELSE 0 END) as '英语',
SUM(CASE `subject` WHEN '政治' THEN score ELSE 0 END) as '政治'
FROM tb_score
GROUP BY userid
注意,SUM() 是为了能够使用GROUP BY根据userid进行分组,因为每一个userid对应的
subject="语文"的记录只有一条,所以SUM() 的值就等于对应那一条记录的score的值。假如
userid ='001' and subject='语文' 的记录有两条,则此时SUM() 的值将会是这两条记录的和,同
理,使用Max()的值将会是这两条记录里面值最大的一个。但是正常情况下,一个user对应一个
subject只有一个分数,因此可以使用SUM()、MAX()、MIN()、AVG()等聚合函数都可以达到行转列
的效果。
使用 IF() 函数实现行转列,参考如下代码:
SELECT userid,
SUM(IF(`subject`='语文',score,0)) as '语文',
SUM(IF(`subject`='数学',score,0)) as '数学',
SUM(IF(`subject`='英语',score,0)) as '英语',
SUM(IF(`subject`='政治',score,0)) as '政治'
FROM tb_score
GROUP BY userid
注意, IF(subject='语文',score,0) 作为条件,即对所有subject='语文'的记录的score字段进
行SUM()、MAX()、MIN()、AVG()操作,如果score没有值则默认为0。
4.1.7 谈谈你对SQL注入的理解
参考答案
SQL注入的原理 是将 SQL代码伪装到输入参数中,传递到服务器解析并执行的一种攻击手法。也就是
说,在一些对SERVER端发起的请求参数中植入一些SQL代码,SERVER端在执行SQL操作时,会拼接对
应参数,同时也将一些SQL注入攻击的“SQL”拼接起来,导致会执行一些预期之外的操作。
举个例子:
比如我们的登录功能,其登录界面包括用户名和密码输入框以及提交按钮 ,登录时需要输入用户名和密
码,然后提交。 此时调用接口/user/login/ 加上参数username、password,首先连接数据库,然后 后
台对请求参数中携带的用户名、密码进行参数校验,即SQL的查询过程。假设正确的用户名和密码为ls
和123456,输入正确的用户名和密码、提交,相当于调用了以下的SQL语句。
SELECT * FROM user WHERE username = 'ls' AND password = '123456'
SQL中会将 #及-- 以后的字符串当做注释处理,如果我们使用 ' or 1=1 # 作为用户名参数,那么服务
端构建的SQL语句就如下:
select * from user where username='' or 1=1 #' and password='123456'
而#会忽略后面的语句,而1=1属于常等型条件,因此这个SQL将查询出所有的登录用户。其实上面的
SQL注入只是在参数层面做了些手脚,如果是引入了一些功能性的SQL那就更危险了,比如上面的登录
功能,如果用户名使用这个 ' or 1=1;delete * from users; # ,那么在";"之后相当于是另外一条新
的SQL,这个SQL是删除全表,是非常危险的操作,因此SQL注入这种还是需要特别注意的。
如何解决SQL注入
而#会忽略后面的语句,而1=1属于常等型条件,因此这个SQL将查询出所有的登录用户。其实上面的
SQL注入只是在参数层面做了些手脚,如果是引入了一些功能性的SQL那就更危险了,比如上面的登录
功能,如果用户名使用这个 ' or 1=1;delete * from users; # ,那么在";"之后相当于是另外一条新
的SQL,这个SQL是删除全表,是非常危险的操作,因此SQL注入这种还是需要特别注意的。
如何解决SQL注入
SQL中会将#及--以后的字符串当做注释处理,如果我们使用 ' or 1=1 # 作为用户名参数,那么服务
端构建的SQL语句就如下:
select * from user where username='' or 1=1 #' and password='123456'
而#会忽略后面的语句,而1=1属于常等型条件,因此这个SQL将查询出所有的登录用户。其实上面的
SQL注入只是在参数层面做了些手脚,如果是引入了一些功能性的SQL那就更危险了,比如上面的登录
功能,如果用户名使用这个 ' or 1=1;delete * from users; # ,那么在";"之后相当于是另外一条新
的SQL,这个SQL是删除全表,是非常危险的操作,因此SQL注入这种还是需要特别注意的。
如何解决SQL注入
严格的参数校验
参数校验就没得说了,在一些不该有特殊字符的参数中提前进行特殊字符校验即可。
SQL预编译
在知道了SQL注入的原理之后,我们同样也了解到MySQL有预编译的功能,指的是 在服务器启动
时,MySQL Client把SQL语句的模板(变量采用占位符进行占位)发送给MySQL服务器,MySQL
服务器对SQL语句的模板进行编译,编译之后根据语句的优化分析对相应的索引进行优化,在最终
绑定参数时把相应的参数传送给MySQL服务器,直接进行执行,节省了SQL查询时间,以及
MySQL服务器的资源,达到一次编译、多次执行的目的,除此之外,还可以防止SQL注入。
具体是怎样防止SQL注入的呢? 实际上当将绑定的参数传到MySQL服务器,MySQL服务器对参数
进行编译,即填充到相应的占位符的过程中,做了转义操作。我们常用的JDBC就有预编译功能,
不仅提升性能,而且防止SQL注入。
4.1.8 将一张表的部分数据更新到另一张表,该如何操作呢?
参考答案
可以采用关联更新的方式,将一张表的部分数据,更新到另一张表内。参考如下代码:
update b set b.col=a.col from a,b where a.id=b.id;
update b set col=a.col from b inner join a on a.id=b.id;
update b set b.col=a.col from b left Join a on b.id = a.id;
4.1.9 WHERE和HAVING有什么区别?
WHERE是一个约束声明,使用WHERE约束来自数据库的数据,WHERE是在结果返回之前起作用的,
WHERE中不能使用聚合函数。
HAVING是一个过滤声明,是在 查询返回结果集以后 对查询结果进行的过滤操作,在HAVING中可以使
用聚合函数。另一方面,HAVING子句中不能使用 **除了分组字段和聚合函数之外的其他字段**。
从性能的角度来说,HAVING子句中如果使用了分组字段作为过滤条件,应该替换成WHERE子句。因为
WHERE可以在执行分组操作和计算聚合函数之前过滤掉不需要的数据,性能会更好。
4.2 索引
4.2.1 说一说你对MySQL索引的理解
索引是一个单独的、存储在磁盘上的数据库结构,包含着对数据表里所有记录的引用指针。使用索引可
以快速找出在某个或多个列中有一特定值的行,所有MySQL列类型都可以被索引,对相关列使用索引是
提高查询操作速度的最佳途径。
索引是在存储引擎中实现的,因此,每种存储引擎的索引都不一定完全相同,并且每种存储引擎也不一
定支持所有索引类型。MySQL中索引的存储类型有两种,即BTREE和HASH,具体和表的存储引擎相
关。MyISAM和InnoDB存储引擎只支持BTREE索引; MEMORY/HEAP存储引擎可以支持HASH和BTREE
索引。
索引的优点主要有以下几条:
通过创建唯一索引,可以保证数据库表中每一行数据的唯一性。
可以大大加快数据的查询速度,这也是创建索引的主要原因。
在实现数据的参考完整性方面,可以加速表和表之间的连接。
在使用分组和排序子句进行数据查询时,也可以显著减少查询中分组和排序的时间。
增加索引也有许多不利的方面,主要表现在如下几个方面:
创建索引和维护索引 要耗费时间,并且随着数据量的增加所耗费的时间也会增加。
索引需要占磁盘空间,除了 数据表占数据空间之外,每一个索引还要占一定的物理空间,如果有大
量的索引,索引文件可能比数据文件更快达到最大文件尺寸。
当对表中的数据进行增加、删除和修改的时候,索引也要动态地维护,这样就降低了数据的维护速
度。
4.2.2 索引有哪几种?
MySQL的索引可以分为以下几类:
MySQL的索引可以分为以下几类:
普通索引和唯一索引
普通索引是MySQL中的基本索引类型,允许 在定义索引的列中插入重复值和空值。
唯一索引要求索引列的值必须唯一,但允许有空值。如果是组合索引,则列值的组合必须唯一。
主键索引是一种特殊的唯一索引,不允许有空值。
单列索引和组合索引
单列索引即一个索引只包含单个列,一个表可以有多个单列索引。
组合索引是指在表的多个字段组合上创建的索引,只有在查询条件中使用了这些字段的左边字段
时,索引才会被使用。使用组合索引时遵循最左前缀集合。
全文索引
全文索引类型为FULLTEXT,在定义索引的列上支持值的全文查找,允许在这些索引列中插入重复
值和空值。全文索引可以在CHAR、VARCHAR或者TEXT类型的列上创建。
空间索引
空间索引是对空间数据类型的字段建立的索引,MySQL中的空间数据类型有4种,分别是
GEOMETRY、POINT、LINESTRING和POLYGON。MySQL使用SPATIAL关键字进行扩展,使得能
够用创建正规索引类似的语法创建空间索引。创建空间索引的列,必须将其声明为NOT NULL,空
间索引只能在存储引擎为MyISAM的表中创建。
4.2.3 如何创建及保存MySQL的索引?
参考答案
MySQL支持多种方法在单个或多个列上创建索引:
在创建表的时候创建索引
使用CREATE TABLE创建表时,除了可以定义列的数据类型,还可以定义主键约束、外键约束或者唯一
性约束,而不论创建哪种约束,在定义约束的同时相当于在指定列上创建了一个索引。创建表时创建索
引的基本语法如下:
CREATE TABLE table_name [col_name data_type]
[UNIQUE|FULLTEXT|SPATIAL] [INDEX|KEY] [index_name] (col_name [length])
[ASC|DESC]
其中,UNIQUE、FULLTEXT和SPATIAL为可选参数,分别表示唯一索引、全文索引和空间索引;INDEX
与KEY为同义词,两者作用相同,用来指定创建索引。
例如,可以按照如下方式,在id字段上使用UNIQUE关键字创建唯一索引:
CREATE TABLE t1 (
id INT NOT NULL,
name CHAR(30) NOT NULL,
UNIQUE INDEX UniqIdx(id)
);
在已存在的表上创建索引
在已经存在的表中创建索引,可以使用ALTER TABLE语句或者CREATEINDEX语句。
ALTER TABLE创建索引的基本语法如下:
ALTER TABLE table_name ADD
[UNIQUE|FULLTEXT|SPATIAL] [INDEX|KEY] [index_name] (col_name[length],...)
[ASC|DESC]
例如,可以按照如下方式,在bookId字段上建立名称为UniqidIdx的唯一索引:
ALTER TABLE book ADD UNIQUE INDEX UniqidIdx (bookId);
CREATE INDEX创建索引的基本语法如下:
CREATE [UNIQUE|FULLTEXT|SPATIAL] INDEX index_name
ON table_name (col_name [length],...) [ASC|DESC]
例如,可以按照如下方式,在bookId字段上建立名称为UniqidIdx的唯一索引:
CREATE UNIQUE INDEX UniqidIdx ON book (bookId);
4.2.4 MySQL怎么判断要不要加索引?
参考答案
建议按照如下的原则来创建索引:
当唯一性是某种数据本身的特征时,指定唯一索引。使用唯一索引需能确保定义的列的数据完整
性,以提高查询速度。
在频繁进行排序或分组(即进行group by或order by操作)的列上建立索引,如果待排序的列有
多个,可以在这些列上建立组合索引。
4.2.5 只要创建了索引,就一定会走索引吗?
参考答案
不一定。
比如,在使用组合索引的时候,如果没有遵从“最左前缀”的原则进行搜索,则索引是不起作用的。
举例,假设在id、name、age字段上已经成功建立了一个名为MultiIdx的组合索引。索引行中按id、
name、age的顺序存放,索引可以搜索id、(id,name)、(id, name, age)字段组合。如果列不构成
索引最左面的前缀,那么MySQL不能使用局部索引,如(age)或者(name,age)组合则不能使用该
索引查询。
4.2.6 如何判断数据库的索引有没有生效?
参考答案:
可以使用EXPLAIN语句查看索引是否正在使用。
举例,假设已经创建了book表,并已经在其year_publication字段上建立了普通索引。执行如下语句::
EXPLAIN SELECT * FROM book WHERE year_publication=1990;
EXPLAIN语句将为我们输出详细的SQL执行信息,其中:
EXPLAIN语句将为我们输出详细的SQL执行信息,其中:
possible_keys行给出了MySQL在搜索数据记录时可选用的各个索引。
key行是MySQL实际选用的索引。
如果possible_keys行和key行都包含year_publication字段,则说明在查询时使用了该索引。
4.2.7 如何评估一个索引创建的是否合理?
建议按照如下的原则来设计索引:
避免**对经常更新的表进行过多的索引,并且索引中的列要尽可能少。应该经常用于查询的字段创建**
索引,但要避免添加不必要的字段。
数据量小的表最好不要使用索引,由于数据较少,查询花费的时间可能比遍历索引的时间还要短,
索引可能不会产生优化效果。
在 条件表达式中 经常用到的不同值较多的列上建立索引, 在不同值很少的列上不要建立索引。比如
在学生表的“性别”字段上只有“男”与“女”两个不同值,因此就无须建立索引,如果建立索引 不但不
会 提高查询效率,反而会严重降低数据更新速度。
当 唯一性是某种数据本身的特征时,指定唯一索引。使用唯一索引需能确保定义的列的数据完整
性,以提高查询速度。
在频繁进行排序或分组(即进行group by或order by操作)的列上建立索引,如果待排序的列有
多个,可以在这些列上建立组合索引。
4.2.8 索引是越多越好吗?
参考答案
索引并非越多越好,一个表中如有大量的索引,不仅占用磁盘空间,还会影响INSERT、DELETE、
UPDATE等语句的性能,因为在表中的数据更改时,索引也会进行调整和更新。
4.2.9 数据库索引失效了怎么办?
参考答案
可以采用以下几种方式,来避免索引失效:
使用 组合索引时,需要遵循“最左前缀”原则;
不在索引列上做任何操作,例如计算、函数、类型转换,会导致索引失效而转向全表扫描;
尽量使用覆盖索引(之访问索引列的查询),减少 select * 覆盖索引能减少回表次数;
MySQL在使用 不等于(!=或者<>)的时候 无法使用索引会导致全表扫描;
LIKE以通配符开头(%abc)MySQL索引会失效变成全表扫描的操作;
字符串不加单引号会导致索引失效(可能发生了索引列的隐式转换);
少用or,用它来连接时会索引失效。
4.2.10 所有的字段都适合创建索引吗?
参考答案
不是。
下列几种情况,是不适合创建索引的:
频繁更新的字段不适合建立索引;
where条件中用不到的字段不适合建立索引;
数据比较少的表不需要建索引;
数据重复且分布比较均匀的的字段不适合建索引,例如性别、真假值;
参与 列计算的列不适合建索引。
4.2.11 说一说索引的实现原理
参考答案:
在MySQL中,索引是在存储引擎层实现的,不同存储引擎对索引的实现方式是不同的,下面我们探讨一
下MyISAM和InnoDB两个存储引擎的索引实现方式。
MyISAM索引实现:
MyISAM引擎使用B+Tree作为索引结构,叶节点的data域存放的是数据记录的地址,MyISAM索引的原
理图如下。这里假设表一共有三列,假设我们以Col1为主键,则上图是一个MyISAM表的主索引
(Primary key)示意。可以看出MyISAM的索引文件仅仅保存数据记录的地址。在MyISAM中,主索引
和辅助索引(Secondary key)在结构上没有任何区别,只是主索引要求key是唯一的,而辅助索引的
key可以重复。
image-20220715114711783
如果我们在Col2上建立一个辅助索引,则此索引的结构如下图所示。同样也是一颗B+Tree,data域保
存数据记录的地址。因此,MyISAM中索引检索的算法为首先按照B+Tree搜索算法搜索索引,如果指定
的Key存在,则取出其data域的值,然后以data域的值为地址,读取相应数据记录。
image-20220715114809280
InnoDB索引实现:
虽然InnoDB也使用B+Tree作为索引结构,但具体实现方式却与MyISAM截然不同。
第一个重大区别是InnoDB的数据文件本身就是索引文件。从上文知道,MyISAM索引文件和数据文件是
分离的,索引文件仅保存数据记录的地址。而在InnoDB中,表数据文件本身就是按B+Tree组织的一个
索引结构,这棵树的叶节点data域保存了完整的数据记录。这个索引的key是数据表的主键,因此
InnoDB表数据文件本身就是主索引。
下图是InnoDB主索引(同时也是数据文件)的示意图,可以看到叶节点包含了完整的数据记录。这种
索引叫做聚集索引。因为InnoDB的数据文件本身要按主键聚集,所以InnoDB要求表必须有主键
(MyISAM可以没有), 如果没有显式指定,则MySQL系统会自动选择一个可以唯一标识数据记录的列
作为主键,如果不存在这种列,则MySQL自动为InnoDB表生成一个隐含字段作为主键,这个字段长度
为6个字节,类型为长整形。
image-20220715115055278
第二个与MyISAM索引的不同是InnoDB的辅助索引data域存储相应记录主键的值而 不是地址。换句话
说,InnoDB的所有辅助索引都引用主键作为data域。下图为定义在Col3上的一个辅助索引。这里以英
文字符的ASCII码作为比较准则。聚集索引这种实现方式使得按主键的搜索十分高效,但是辅助索引搜
索需要检索两遍索引:首先检索辅助索引获得主键,然后用主键到主索引中检索获得记录。
image-20220715115148293
了解不同存储引擎的索引实现方式对于正确使用和优化索引都非常有帮助,例如知道了InnoDB的索引
实现后,就很容易明白为什么不建议使用过长的字段作为主键,因为所有辅助索引都引用主索引,过长
的主索引会令辅助索引变得过大。再例如,用非单调的字段作为主键在InnoDB中不是个好主意,因为
InnoDB数据文件本身是一颗B+Tree,非单调的主键会造成在插入新记录时数据文件为了维持B+Tree的
特性而频繁的分裂调整,十分低效,而使用自增字段作为主键则是一个很好的选择。
4.2.12 介绍一下数据库索引的重构过程
参考答案
什么时候需要重建索引呢?
表上频繁发生update,delete操作;
表上发生了alter table ..move操作(move操作导致了rowid变化)。
怎么判断索引是否应该重建?
一般看索引是否倾斜的严重,是否浪费了空间,对索引进行结构分析:
analyze index index_name validate structure;
在相同的session中查询index_stats表:
select height,DEL_LF_ROWS/LF_ROWS from index_stats;
当查询的height>=4(索引的深度,即从根到叶节点的高度)或DEL_LF_ROWS/LF_ROWS>0.2的情
况下,就应该考虑重建该索引。
如何重建索引?
drop原索引,然后再创建索引:
drop index index_name;
create index index_name on table_name (index_column);
这种方式相当耗时,一般不建议使用。
直接重建索引:
alter index indexname rebuild;
alter index indexname rebuild online;
此方法较快,建议使用。
rebuild是快速重建索引的一种有效的办法,因为它是一种使用现有索引项来重建新索引的方法。如果重
建索引时有其他用户在对这个表操作,尽量使用带online参数来最大限度的减少索引重建时将会出现的
任何加锁问题。 由于 新旧索引在建立时同时存在,因此,使用这种重建方法需要有额外的磁盘空间可供
临时使用,当索引建完后把老索引删除,如果没有成功,也不会影响原来的索引。利用这种办法可以用
来将一个索引移到新的表空间。
rebuild重建索引的过程:
Rebuild 以index fast full scan或 table full scan方式(采用那种方式取决于cost)读取原索引中的
数据来构建一个新的索引,重建过程中有排序操作, rebuild online执行表扫描获取数据,重建过
程中有排序的操作;
Rebuild 会阻塞DML操作,rebuild online不会阻塞DML操作;
rebuild online时系统会产生一个SYS_JOURNAL_xxx的IOT类型的系统临时日志表,所有rebuild
online时索引的变化都记录在这个表中, 当新的索引创建完成后,把这个表的记录维护到新的索引
中去,然后drop掉旧的索引,rebuild online就完成了。
重建索引过程中的注意事项:
执行rebuild操作时,需要检查表空间是否足够;
虽然说rebuild online操作允许DML操作,但还是建议在业务不繁忙时间段进行;
Rebuild操作会产生大量Redo Log;
4.2.13 MySQL的索引为什么用B+树?
参考答案
B+树 由B树和索引顺序访问方法演化而来,它是为磁盘或其他直接存取辅助设备设计的一种平衡查找
树,在B+树中,所有记录节点都是按键值的大小顺序存放在同一层的叶子节点,各叶子节点通过指针进
行链接。如下图
image-20220715141203786
B+树索引在 数据库中的一个特点就是高扇出性,例如在InnoDB存储引擎中,每个页的大小为16KB。在
数据库中,B+树的高度一般都在2~4层,这意味着查找某一键值最多只需要2到4次IO操作,这还不
错。因为现在一般的磁盘每秒至少可以做100次IO操作, 2~4次的IO操作意味着查询时间只需0.02~
0.04秒。
4.2.14 联合索引的存储结构是什么,它的有效方式是什么?
参考答案
从本质上来说,联合索引还是一棵B+树,不同的是联合索引的键值数量不是1,而是大于等于2,参考
下图。另外,只有在查询条件中使用了这些字段的左边字段时,索引才会被使用,所以使用联合索引时
遵循最左前缀集合。
image-20220715141416654
4.2.15 MySQL的Hash索引和B树索引有什么区别?
参考答案:'
hash索引底层就是hash表,进行查找时,调用一次hash函数就可以获取到相应的键值,之后进行回表
查询获得实际数据。 B+树底层实现是多路平衡查找树,对于每一次的查询都是从根节点出发,查找到叶
子节点方可以获得所查键值,然后根据查询判断是否需要回表查询数据。它们有以下的不同:
hash索引 进行等值查询更快(一般情况下),但是却无法进行范围查询。因为在hash索引中经过
hash函数建立索引之后,索引的顺序与原顺序无法保持一致,不能支持范围查询。而B+树的的所
有节点皆遵循(左节点小于父节点,右节点大于父节点,多叉树也类似),天然支持范围。
hash索引不支持 使用索引进行排序,原理同上。
hash索引不支持 模糊查询以及多列索引的最左前缀匹配,原理也是因为hash函数的不可预测。
hash索引 任何时候都避免不了回表查询数据,而B+树在符合某些条件(聚簇索引,覆盖索引等)的
时候可以只通过索引完成查询。
hash索引 虽然在等值查询上较快,但是不稳定,性能不可预测,当某个键值存在大量重复的时
候,发生hash碰撞,此时效率可能极差。而B+树的查询效率比较稳定,对于所有的查询都是从根
节点到叶子节点,且树的高度较低。
因此,在大多数情况下,直接选择B+树索引可以获得稳定且较好的查询速度。而不需要使用hash索
引。
4.2.16 聚簇索引和非聚簇索引有什么区别?
参考答案
在InnoDB存储引擎中,可以将B+树索引分为聚簇索引和辅助索引(非聚簇索引)。无论是何种索引,
每个页的大小都为16KB,且不能更改。
聚簇索引 是根据主键创建的一棵B+树,聚簇索引 的叶子节点 存放了表中的所有记录。辅助索引 是根据索
引键创建的一棵B+树,与聚簇索引不同的是,其叶子节点仅存放索引键值,以及该索引键值指向的主
键。也就是说,如果通过 辅助索引 来查找数据,那么当找到辅助索引的叶子节点后,很有可能还需要根
据主键值查找聚簇索引来得到数据,这种查找方式又被称为书签查找。因为辅助索引不包含行记录的所
有数据,这就意味着每页可以存放更多的键值,因此其高度一般都要小于聚簇索引。
4.2.17 什么是联合索引?
:
参考答案:
联合索引是指对表上的多个列进行索引,联合索引的创建方法与单个索引创建的方法一样,不同之处仅
在于有多个索引列。从本质上来说,联合索引还是一棵B+树,不同的是联合索引的键值数量不是1,而
是大于等于2,参考下图。另外,只有在查询条件中使用了这些字段的左边字段时,索引才会被使用,
所以使用联合索引时遵循最左前缀集合。
image-20220715142428426
4.2.18 select in语句中如何使用索引?
参考答案
索引是否起作用,主要取决于字段类型:
如果字段类型为字符串,需要给in查询中的数值与字符串值都需要添加引号,索引才能起作用。
如果字段类型为int,则in查询中的值不需要添加引号,索引也会起作用。
IN的字段,在联合索引中,按以上方法,也会起作用。
4.2.19 模糊查询语句中如何使用索引?
参考答案:
在MySQL中模糊查询 mobile like ‘%8765’ ,这种情况是不能使用 mobile 上的索引的,那么如果需
要根据手机号码后四位进行模糊查询,可以用一下方法进行改造。
我们可以加入冗余列(MySQL5.7之后加入了虚拟列,使用虚拟列更合适,思路相同),比如
mobile_reverse,内部存储为 mobile 的倒叙文本,如 mobile为17312345678,那么 mobile_reverse
存储 87654321371,为 mobile_reverse 列建立索引,查询中使用语句 mobile_reverse like
reverse(’%5678’) 即可。
reverse 是 MySQL 中的反转函数,这条语句相当于 mobile_reverse like ‘8765%’ ,这种语句是可以使
用索引的。
4.3 事务
4.3.1 说一说你对数据库事务的了解
参考答案
事务可由一条非常简单的SQL语句组成,也可以由一组复杂的SQL语句组成。在事务中的操作,要么都
执行修改,要么都不执行,这就是事务的目的,也是事务模型区别于文件系统的重要特征之一。
事务需遵循ACID四个特性:
A(atomicity),原子性。原子性 指整个数据库事务是不可分割的工作单位。只有使事务中所有的
数据库操作都执行成功,整个事务的执行才算成功。事务中任何一个SQL语句执行失败,那么已经执行成功的SQL语句也必须撤销,数据库状态应该退回到执行事务前的状态。
C(consistency),一致性。一致性指事务将数据库从一种状态转变为另一种一致的状态。在事
务开始之前和事务结束以后,数据库的完整性约束没有被破坏。
I(isolation),隔离性。事务的隔离性要求每个读写事务的对象与 其他事务的操作对象能相互分
离,即该事务提交前对其他事务都不可见,这通常使用锁来实现。
D(durability) ,持久性。事务一旦提交,其结果就是永久性的,即使发生宕机等故障,数据库
也能将数据恢复。持久性保证的是事务系统的高可靠性,而不是高可用性。
事务可以分为以下几种类型:
扁平事务:是事务类型中最简单的一种,而在实际生产环境中,这可能是使用最为频繁的事务。在
扁平事务中,所有操作都处于同一层次,其由BEGIN WORK开始,由COMMIT WORK或
ROLLBACK WORK结束。处于之间的操作是原子的,要么都执行,要么都回滚。
带有保存点的扁平事务:除了支持扁平事务支持的操作外,允许在事务执行过程中回滚到同一事务
中较早的一个状态,这是因为可能某些事务在执行过程中出现的错误并不会对所有的操作都无效,
放弃整个事务不合乎要求,开销也太大。保存点(savepoint)用来通知系统应该记住事务当前的
状态,以便以后发生错误时,事务能回到该状态。
链事务:可视为保存点模式的一个变种。链事务的思想是:在提交一个事务时,释放不需要的数据
对象,将必要的处理上下文隐式地传给下一个要开始的事务。注意,提交事务操作和开始下一个事
务操作将合并为一个原子操作。这意味着下一个事务将看到上一个事务的结果,就好像在一个事务
中进行的
嵌套事务:是一个层次结构框架。有一个顶层事务(top-level transaction)控制着各个层次的事
务。顶层事务之下嵌套的事务被称为子事务(subtransaction),其控制每一个局部的变换。
分布式事务:通常是一个在分布式环境下运行的扁平事务,因此需要根据数据所在位置访问网络中
的不同节点。对于分布式事务,同样需要满足ACID特性,要么都发生,要么都失效。
对于MySQL的InnoDB存储引擎来说,它支持扁平事务、带有保存点的扁平事务、链事务、分布式事
务。对于嵌套事务,MySQL数据库并不是原生的,因此对于有并行事务需求的用户来说MySQL就无能
为力了,但是用户可以通过带有保存点的事务来模拟串行的嵌套事务。
4.3.2 事务有哪几种类型,它们之间有什么区别?
事务可以分为以下几种类型:
扁平事务:是事务类型中最简单的一种,而在实际生产环境中,这可能是使用最为频繁的事务。在
扁平事务中,所有操作都处于同一层次,其由BEGIN WORK开始,由COMMIT WORK或
ROLLBACK WORK结束。处于之间的操作是原子的,要么都执行,要么都回滚。
带有保存点的扁平事务:除了支持扁平事务支持的操作外,允许在事务执行过程中回滚到同一事务
中较早的一个状态,这是因为可能某些事务在执行过程中出现的错误并不会对所有的操作都无效,
放弃整个事务不合乎要求,开销也太大。保存点(savepoint)用来通知系统应该记住事务当前的
状态,以便以后发生错误时,事务能回到该状态。
链事务:可视为保存点模式的一个变种。链事务的思想是:在提交一个事务时,释放不需要的数据
对象,将必要的处理上下文隐式地传给下一个要开始的事务。注意,提交事务操作和开始下一个事
务操作将合并为一个原子操作。这意味着下一个事务将看到上一个事务的结果,就好像在一个事务
中进行的
嵌套事务:是一个层次结构框架。有一个顶层事务(top-level transaction)控制着各个层次的事
务。顶层事务之下嵌套的事务被称为子事务(subtransaction),其控制每一个局部的变换。
分布式事务:通常是一个在分布式环境下运行的扁平事务,因此需要根据数据所在位置访问网络中
的不同节点。对于分布式事务,同样需要满足ACID特性,要么都发生,要么都失效。
对于MySQL的InnoDB存储引擎来说,它支持扁平事务、带有保存点的扁平事务、链事务、分布式事
务。对于嵌套事务,MySQL数据库并不是原生的,因此对于有并行事务需求的用户来说MySQL就无能
为力了,但是用户可以通过带有保存点的事务来模拟串行的嵌套事务。
4.3.3 MySQL的ACID特性分别是怎么实现的?
参考答案:
原子性实现原理:
实现原子性的关键,是当事务回滚时能够撤销所有已经成功执行的sql语句。InnoDB实现回滚靠的是
undo log,当事务对数据库进行修改时,InnoDB会生成对应的undo log。如果事务执行失败或调用了
rollback,导致事务需要回滚,便可以利用undo log中的信息将数据回滚到修改之前的样子。
undo log属于逻辑日志,它记录的是sql执行相关的信息。当发生回滚时,InnoDB会根据undo log的内
容做与之前相反的工作。对于insert,回滚时会执行delete。对于delete,回滚时会执行insert。对于
update,回滚时则会执行相反的update,把数据改回去。
持久性实现原理:
InnoDB作为MySQL的存储引擎,数据是存放在磁盘中的,但如果每次读写数据都需要磁盘IO,效率会
很低。为此,InnoDB提供了缓存(Buffer Pool),Buffer Pool中包含了磁盘中部分数据页的映射,作为
访问数据库的缓冲。当从数据库读取数据时,会首先从Buffer Pool中读取,如果Buffer Pool中没有,
则从磁盘读取后放入Buffer Pool。当向数据库写入数据时,会首先写入Buffer Pool,Buffer Pool中修
改的数据会定期刷新到磁盘中(这一过程称为刷脏)。
Buffer Pool的使用大大提高了读写数据的效率,但是也带了新的问题:如果MySQL宕机,而此时
Buffer Pool中修改的数据还没有刷新到磁盘,就会导致数据的丢失,事务的持久性无法保证。
于是,redo log被引入来解决这个问题。当数据修改时,除了修改Buffer Pool中的数据,还会在redo
log记录这次操作。当事务提交时,会调用fsync接口对redo log进行刷盘。如果MySQL宕机,重启时可
以读取redo log中的数据,对数据库进行恢复。redo log采用的是WAL(Write-ahead logging,预写式
日志),**所有修改先写入日志,再更新到Buffer Pool,保证了数据不会因MySQL宕机而丢失,从而满**
足了持久性要求。
既然redo log也需要在事务提交时将日志写入磁盘,为什么它比直接将Buffer Pool中修改的数据写入磁
盘(即刷脏)要快呢?主要有以下两方面的原因:
刷脏是随机IO,因为每次修改的数据位置随机,但写redo log是追加操作,属于顺序IO。
刷脏是以数据页(Page)为单位的,MySQL默认页大小是16KB,一个Page上一个小修改都要整
页写入。而redo log中只包含真正需要写入的部分,无效IO大大减少。
隔离性实现原理:
隔离性追求的是并发情形下事务之间互不干扰。简单起见,我们主要考虑最简单的读操作和写操作(加锁
读等特殊读操作会特殊说明),那么隔离性的探讨,主要可以分为两个方面。
第一方面,(一个事务)写操作对(另一个事务)写操作的影响:锁机制保证隔离性。
隔离性要求同一时刻只能有一个事务对数据进行写操作,InnoDB通过锁机制来保证这一点。锁机制的
基本原理可以概括为:事务在修改数据之前,需要先获得相应的锁。获得锁之后,事务便可以修改数
据。该事务操作期间,这部分数据是锁定的,其他事务如果需要修改数据,需要等待当前事务提交或回
滚后释放锁。
按照粒度, 锁可以分为表锁、行锁以及其他位于二者之间的锁。表锁在操作数据时会锁定整张表,并发
性能较差。**行锁则只锁定需要操作的数据,并发性能好。但是由于加锁本身需要消耗资源,因此在锁定**
数据较多情况下使用表锁可以节省大量资源。
MySQL中不同的存储引擎支持的锁是不一样的,例如MyIsam只支持表锁,而InnoDB同时支持表锁和行锁,且出于性能考虑,绝大多数情况下使用的都是行锁。
第二方面,(一个事务)写操作对(另一个事务)读操作的影响:MVCC保证隔离性。
InnoDB默认的隔离级别是RR(REPEATABLE READ), RR解决脏读、不可重复读、幻读等问题,使用
的是MVCC。MVCC全称Multi-Version Concurrency Control,即多版本的并发控制协议。它最大的优
点是读不加锁,因此读写不冲突,并发性能好。InnoDB实现MVCC,多个版本的数据可以共存,主要基
于以下技术及数据结构:
隐藏列:InnoDB中每行数据都有隐藏列,隐藏列中包含了本行数据的事务id、指向undo log的指
针等。
基于undo log的版本链:每行数据的隐藏列中包含了指向undo log的指针,而每条undo log也会
指向更早版本的undo log,从而形成一条版本链。
ReadView:通过隐藏列和版本链,MySQL可以将数据恢复到指定版本。但是具体要恢复到哪个版
本,则需要根据ReadView来确定。 所谓ReadView,是指事务(记做事务A)在某一时刻给整个事
务系统(trx_sys)打快照,之后再进行读操作时,会将读取到的数据中的事务id与trx_sys快照比
较,从而判断数据对该ReadView是否可见,即对事务A是否可见。
一致性实现原理:
可以说,一致性是事务追求的最终目标。前面提到的原子性、持久性和隔离性,都是为了保证数据库状
态的一致性。此外,除了数据库层面的保障,一致性的实现也需要应用层面进行保障。实现一致性的措
施包括:
保证原子性、持久性和隔离性,如果这些特性无法保证,事务的一致性也无法保证。
数据库本身提供保障,例如不允许向整形列插入字符串值、字符串长度不能超过列的限制等。
应用层面进行保障,例如如果转账操作只扣除转账者的余额,而没有增加接收者的余额,无论数据
库实现的多么完美,也无法保证状态的一致。
4.3.4 谈谈MySQL的事务隔离级别
参考答案
SQL 标准定义了四种隔离级别,这四种隔离级别分别是:
读未提交(READ UNCOMMITTED);
读提交 (READ COMMITTED);
可重复读 (REPEATABLE READ);
串行化 (SERIALIZABLE)。
事务隔离是为了解决脏读、不可重复读、幻读问题,下表展示了 4 种隔离级别对这三个问题的解决程
度:
image-20220715151035168
上述4种隔离级别MySQL都支持,并且InnoDB存储引擎默认的支持隔离级别是REPEATABLE READ,但
是与标准SQL不同的是,InnoDB存储引擎在REPEATABLE READ事务隔离级别下,使用Next-Key Lock
的锁算法,因此避免了幻读的产生。所以,InnoDB存储引擎在默认的事务隔离级别下已经能完全保证
事务的隔离性要求,即达到SQL标准的SERIALIZABLE隔离级别。
扩展阅读
并发情况下,读操作可能存在的三类问题:
脏读:当前事务(A)中可以读到其他事务(B)未提交的数据(脏数据),这种现象是脏读。
不可重复读:在事务A中先后两次读取同一个数据,两次读取的结果不一样,这种现象称为不可重
复读。脏读与不可重复读的区别在于:前者读到的是其他事务未提交的数据,后者读到的是其他事
务已提交的数据。
幻读:在事务A中按照某个条件先后两次查询数据库,两次查询结果的条数不同,这种现象称为幻
读。不可重复读与幻读的区别可以通俗的理解为:前者是数据变了,后者是数据的行数变了。
4.3.5 MySQL的事务隔离级别是怎么实现的?
InnoDB支持四种隔离级别,每种级别解决掉的问题如下表:
image-20220715153801824
这四种隔离级别的实现机制如下::
READ UNCOMMITTED & READ COMMITTED:
通过Record Lock算法实现了行锁,但READ UNCOMMITTED允许读取未提交数据,所以存在脏读
问题。而READ COMMITTED允许读取提交数据,所以不存在脏读问题,但存在不可重复读问题。
REPEATABLE READ:
使用Next-Key Lock算法实现了行锁,并且不允许读取已提交的数据,所以解决了不可重复读的问
题。另外,该算法包含了间隙锁,会锁定一个范围,因此也解决了幻读的问题。
SERIALIZABLE:
对每个SELECT语句后自动加上LOCK IN SHARE MODE,即为每个读取操作加一个共享锁。因此在
这个事务隔离级别下,读占用了锁,对一致性的非锁定读不再予以支持。
4.3.6 事务可以嵌套吗?
参考答案
可以,因为嵌套事务也是众多事务分类中的一种,它是一个层次结构框架。有一个顶层事务控制着各个
层次的事务,顶层事务之下嵌套的事务被称为子事务,它控制每一个局部的变换。
需要注意的是,MySQL数据库不支持嵌套事务。
4.3.7 如何实现可重复读?
参考答案
MySQL的InnoDB引擎,在默认的REPEATABLE READ的隔离级别下,实现了可重复读,同时也解决了幻
读问题。它使用Next-Key Lock算法实现了行锁,并且不允许读取已提交的数据,所以解决了不可重复
读的问题。另外,该算法包含了间隙锁,会锁定一个范围,因此也解决了幻读的问题。
4.3.8 如何解决幻读问题?
参考答案
MySQL的InnoDB引擎,在默认的REPEATABLE READ的隔离级别下,实现了可重复读,同时也解决了幻
读问题。它使用Next-Key Lock算法实现了行锁,并且不允许读取已提交的数据,所以解决了不可重复
读的问题。另外,该算法包含了间隙锁,会锁定一个范围,因此也解决了幻读的问题。
4.3.9 MySQL事务如何回滚?
在MySQL默认的配置下,事务都是自动提交和回滚的。当显示地开启一个事务时,可以使用ROLLBACK
语句进行回滚。该语句有两种用法:
语句进行回滚。该语句有两种用法:
ROLLBACK:要使用这个语句的最简形式,只需发出ROLLBACK。同样地,也可以写为ROLLBACK
WORK,但是二者几乎是等价的。回滚会结束用户的事务,并撤销正在进行的所有未提交的修改。
ROLLBACK TO [SAVEPOINT] identifier : 这个语句与SAVEPOINT命令一起使用。可以把事务回滚
到标记点,而不回滚在此标记点之前的任何工作。
4.4 锁
4.4.1 了解数据库的锁吗?
参考答案
锁是数据库系统 区别于文件系统的一个关键特性**,锁机制用于管理对共享资源的并发访问。**下面我们以
MySQL数据库的InnoDB引擎为例,来说明锁的一些特点。
锁的类型:
InnoDB存储引擎实现了如下两种标准的行级锁:
共享锁(S Lock),允许 事务 读一行数据。
排他锁(X Lock),允许事务 删除或更新一行数据。
如果一个 事务T1 已经获得了行r的共享锁,那么另外的事务T2可以立即获得行r的共享锁,因为读取并没
有改变行r的数据,称这种情况为锁兼容。但若有其他的事务T3想获得行r的排他锁,则其必须等待事务
T1、T2释放行r上的共享锁,这种情况称为锁不兼容。下图显示了共享锁和排他锁的兼容性,可以发现X
锁与任何的锁都不兼容,而S锁仅和S锁兼容。需要特别注意的是,S和X锁都是行锁,兼容是指对同一记
录(row)锁的兼容性情况。
image-20220715154519771
锁的粒度:
InnoDB存储引擎支持多粒度锁定,这种锁定允许事务在行级上的锁和表级上的锁同时存在。为了支持
在不同粒度上进行加锁操作,InnoDB存储引擎支持一种额外的锁方式,称之为意向锁。意向锁是将锁
定的对象分为多个层次,意向锁意味着事务希望在更细粒度上进行加锁。
InnoDB存储引擎支持意向锁设计比较简练,其意向锁即为表级别的锁。设计目的主要是为了在一个事
务中揭示下一行将被请求的锁类型。其支持两种意向锁:
意向共享锁(IS Lock),事务想要获得一张表中某几行的共享锁。
意向排他锁(IX Lock),事务想要获得一张表中某几行的排他锁。
由于InnoDB存储引擎支持的是行级别的锁,因此意向锁其实不会阻塞除全表扫以外的任何请求。故表
级意向锁与行级锁的兼容性如下图所示。
image-20220715154703860
锁的算法:
InnoDB存储引擎有3种行锁的算法,其分别是:
Record Lock: 单个行记录上的锁。
Gap Lock: 间隙锁,锁定一个范围,但不包含记录本身。
Next-Key Lock∶Gap Lock+Record Lock,锁定一个范围,并且锁定记录本身。
Record Lock总是会去锁住索引记录,如果InnoDB存储引擎表在建立的时候没有设置任何一个索引,那
么这时InnoDB存储引擎会使用隐式的主键来进行锁定。
Next-Key Lock是结合了Gap Lock和Record Lock的一种锁定算法,在Next-Key Lock算法下,InnoDB对于行的查询都是采用这种锁定算法。采用Next-Key Lock的锁定技术称为Next-Key Locking,其设计的目的是为了解决Phantom Problem(幻读)。而利用这种锁定技术,锁定的不是单个值,而是一个范围,是谓词锁(predict lock)的一种改**进。**
关于死锁:
死锁 是指两个或两个以上的事务在执行过程中,因争夺锁资源而造成的一种互相等待的现象。若无外力
作用,事务都将无法推进下去。
解决死锁问题最简单的一种方法是超时,即当两个事务互相等待时,当一个等待时间超过设置的某一阈
值时,其中一个事务进行回滚,另一个等待的事务就能继续进行。
除了超时机制,当前数据库还都普遍采用wait-for graph(等待图)的方式来进行死锁检测。较之超时
的解决方案,这是一种更为主动的死锁检测方式。InnoDB存储引擎也采用的这种方式。wait-for graph
要求数据库保存以下两种信息:
锁的信息链表;
事务等待链表;
通过上述链表可以构造出一张图,而在这个图中若存在回路,就代表存在死锁,因此资源间相互发生等
待。这是一种较为主动的死锁检测机制,在每个事务请求锁并发生等待时都会判断是否存在回路,若存
在则有死锁,通常来说InnoDB存储引擎选择回滚undo量最小的事务。
锁的升级:
锁升级(Lock Escalation)是指将当前锁的粒度降低。举例来说,数据库可以把一个表的1000个行锁
升级为一个页锁,或者 将页锁升级为表锁。
InnoDB存储引擎不存在锁升级的问题。因为其 不是根据每个记录来产生行锁的,相反,其根据每个事
务访问的每个页对锁进行管理的,采用的是位图的方式。因此不管一个事务锁住页中一个记录还是多个
记录,其开销通常都是一致的。
4.4.2 介绍一下间隙锁
参考答案
InnoDB存储引擎有3种行锁的算法, 间隙锁(Gap Lock)是其中之一。间隙锁用于锁定一个范围,但
不包含记录本身。它的作用是为了阻止多个事务将记录插入到同一范围内,而这会导致幻读问题的产
生。
4.4.3 InnoDB中行级锁是怎么实现的?
参考答案
InnoDB行级锁 是通过 给索引上的索引项 加锁来实现的。只有通过索引条件检索数据,InnoDB才使用行
级锁,否则,InnoDB将使用表锁。
当表中锁定其中的某几行时, 不同的事务 可以 使用不同的索引锁定不同的行。另外,不论使用主键索
引、唯一索引还是普通索引,InnoDB都会使用行锁来对数据加锁。
4.4.4 数据库在什么情况下会发生死锁?
死锁是指两个或两个以上的事务在执行过程中,因争夺锁资源而造成的一种互相等待的现象。若无外力
作用,事务都将无法推进下去。下图演示了死锁的一种经典的情况,即A等待B、B等待A,这种死锁问
题被称为AB-BA死锁。
image-20220715160128094
4.4.5 说说数据库死锁的解决办法
解决死锁问题最简单的一种方法是超时,即当两个事务互相等待时,当一个等待时间超过设置的某一阈
值时,其中一个事务进行回滚,另一个等待的事务就能继续进行。
除了超时机制,当前数据库还都普遍采用wait-for graph(等待图)的方式来进行死锁检测。较之超时
的解决方案,这是一种更为主动的死锁检测方式。InnoDB存储引擎也采用的这种方式。wait-for graph
要求数据库保存以下两种信息:
锁的信息链表;
事务等待链表;
通过上述链表可以构造出一张图,而在这个图中若存在回路,就代表存在死锁,因此资源间相互发生等
待。这是一种较为主动的死锁检测机制,在每个事务请求锁并发生等待时都会判断是否存在回路,若存
在则有死锁,通常来说InnoDB存储引擎选择回滚undo量最小的事务。
4.5 优化
4.5.1 说一说你对数据库优化的理解
参考答案
MySQL数据库优化是多方面的,原则是 减少系统的瓶颈,减少资源的占用,增加系统的反应速度。例
如, 通过优化文件系统,提高磁盘I\O的读写速度;通过优化操作系统调度策略,提高MySQL在高负荷
情况下的负载能力;优化表结构、索引、查询语句等使查询响应更快。
针对查询, 我们可以通过使用索引、使用连接代替子查询的方式来提高查询速度。
针对慢查询, 我们可以通过分析慢查询日志,来发现引起慢查询的原因,从而有针对性的进行优化。
针对插入,我们可以通过禁用索引、禁用检查等方式来提高插入速度,在插入之后再启用索引和检查。
针对数据库结构,我们可以通过将字段很多的表拆分成多张表、增加中间表、增加冗余字段等方式进行
优化。
4.5.2 该如何优化MySQL的查询?
参考答案:
使用索引:
如果查询时 没有使用索引,查询语句将扫描表中的所有记录。在数据量大的情况下,这样查询的速度会
很慢。如果使用索引进行查询,查询语句可以根据索引快速定位到待查询记录,从而减少查询的记录
数,达到提高查询速度的目的。
索引 可以提高查询的速度,但并不是使用带有索引的字段查询时索引都会起作用。有几种特殊情况,在
这些情况下有可能使用带有索引的字段查询时索引并没有起作用。
使用LIKE关键字的查询语句
在使用LIKE关键字进行查询的查询语句中,如果匹配字符串的第一个字符为“%”,索引不会起作
用。只有“%”不在第一个位置,索引才会起作用。
使用 多列索引的查询语句MySQL可以为多个字段创建索引。一个索引可以包括16个字段。对于多列索引,只有查询条件中使用了这些字段中的第1个字段时索引才会被使用。
使用OR关键字的查询语句
查询语句的查询条件中只 有OR关键字,且OR前后的两个条件中的列都是索引时,查询中才使用索
引。否则,查询将不使用索引。
优化子查询:
使用子查询可以进行SELECT语句的嵌套查询,即一个SELECT查询的结果作为另一个SELECT语句的条
件。子查询可以一次性完成很多逻辑上需要多个步骤才能完成的SQL操作。
子查询虽然可以使查询语句很灵活,但执行效率不高。执行子查询时,MySQL需要为内层查询语句的查
询结果建立一个临时表。然后外层查询语句从临时表中查询记录。查询完毕后,再撤销这些临时表。因
此,子查询的速度会受到一定的影响。如果查询的数据量比较大,这种影响就会随之增大。
在MySQL中,可以使用连接(JOIN)查询来替代子查询。连接查询不需要建立临时表,其速度比子查
询要快,如果查询中使用索引,性能会更好。
4.5.3 怎样插入数据才能更高效?
参考答案
影响插入速度的主要是索引、唯一性校验、一次插入记录条数等。针对这些情况,可以分别进行优化。
对于
禁用索引
对于非空表,插入记录时,MySQL会根据表的索引对插入的记录建立索引。如果插入大量数据,
建立索引会降低插入记录的速度。为了解决这种情况,可以在插入记录之前禁用索引,数据插入完
毕后再开启索引。对于空表批量导入数据,则不需要进行此操作,因为MyISAM引擎的表是在导入
数据之后才建立索引的。
禁用唯一性检查
插入数据时,MySQL会对插入的记录进行唯一性校验。这种唯一性校验也会降低插入记录的速
度。为了降低这种情况对查询速度的影响,可以在插入记录之前禁用唯一性检查,等到记录插入完
毕后再开启。
使用批量插入
插入多条记录时,可以使用一条INSERT语句插入一条记录,也可以使用一条INSERT语句插入多条
记录。使用一条INSERT语句插入多条记录的情形如下,而这种方式的插入速度更快。
使用LOAD DATA INFILE批量导入
当需要批量导入数据时,如果能用LOAD DATA INFILE语句,就尽量使用。因为LOAD DATA INFILE
语句导入数据的速度比INSERT语句快。
对于InnoDB引擎的表,常见的优化方法如下:
禁用唯一性检查
插入数据之前执行 set unique_checks=0 来禁止对唯一索引的检查,数据导入完成之后再运行
set unique_checks=1 。这个和MyISAM引擎的使用方法一样。
禁用外键检查
插入数据之前执行禁止对外键的检查,数据插入完成之后再恢复对外键的检查。
禁用自动提交
插入数据之前禁止事务的自动提交,数据导入完成之后,执行恢复自动提交操作。
4.5.4 表中包含几千万条数据该怎么办?
建议按照如下顺序进行优化:
优化SQL和索引;
增加缓存,如memcached、redis;
读写分离,可以采用主从复制,也可以采用主主复制;
使用MySQL自带的分区表,这对应用是透明的,无需改代码,但SQL语句是要针对分区表做优化
的;
做垂直拆分,即根据模块的耦合度,将一个大的系统分为多个小的系统;
做水平拆分,要选择一个合理的sharding key,为了有好的查询效率,表结构也要改动,做一定的
冗余,应用也要改,sql中尽量带sharding key,将数据定位到限定的表上去查,而不是扫描全部
的表
4.5.5 MySQL的慢查询优化有了解吗?
参考答案
优化MySQL的慢查询,可以按照如下步骤进行:
开启慢查询日志:
MySQL中慢查询日志默认是关闭的,可以通过配置文件my.ini或者myf中的log-slow-queries选项打
开,也可以在MySQL服务启动的时候使用 --log-slow-queries[=file_name] 启动慢查询日志。
启动慢查询日志时,需要在my.ini或者myf文件中配置long_query_time选项指定记录阈值,如果某
条查询语句的查询时间超过了这个值,这个查询过程将被记录到慢查询日志文件中。
分析慢查询日志:
直接分析mysql慢查询日志,利用explain关键字可以模拟优化器执行SQL查询语句,来分析sql慢查询
语句。
常见慢查询优化:
索引没起作用的情况
在使用LIKE关键字进行查询的查询语句中,如果匹配字符串的第一个字符为“%”,索引不会起
作用。只有“%”不在第一个位置,索引才会起作用。
MySQL可以为多个字段创建索引。一个索引可以包括16个字段。对于多列索引,只有查询条
件中使用了这些字段中的第1个字段时索引才会被使用。
查询语句的查询条件中只有OR关键字,且OR前后的两个条件中的列都是索引时,查询中才
使用索引。否则,查询将不使用索引。
优化数据库结构
对于字段比较多的表,如果有些字段的使用频率很低,可以将这些字段分离出来形成新表。
因为当一个表的数据量很大时,会由于使用频率低的字段的存在而变慢。
对于需要经常联合查询的表,可以建立中间表以提高查询效率。通过建立中间表,把需要经
常联合查询的数据插入到中间表中,然后将原来的联合查询改为对中间表的查询,以此来提
高查询效率
分解关联查询
很多高性能的应用都会对关联查询进行分解,就是可以对每一个表进行一次单表查询,然后将查询
结果在应用程序中进行关联,很多场景下这样会更高效。
优化LIMIT分页
当偏移量非常大的时候,例如可能是limit 10000,20这样的查询,这是mysql需要查询10020条然
后只返回最后20条,前面的10000条记录都将被舍弃,这样的代价很高。优化此类查询的一个最
简单的方法是尽可能的使用索引覆盖扫描,而不是查询所有的列。然后根据需要做一次关联操作再
返回所需的列。对于偏移量很大的时候这样做的效率会得到很大提升。
4.5.6 说一说你对explain的了解
MySQL中提供了EXPLAIN语句和DESCRIBE语句,用来分析查询语句,EXPLAIN语句的基本语法如下
EXPLAIN [EXTENDED] SELECT select_options
使用EXTENED关键字,EXPLAIN语句将产生附加信息。执行该语句,可以分析EXPLAIN后面SELECT语
句的执行情况,并且能够分析出所查询表的一些特征。下面对查询结果进行解释
id:SELECT识别符。这是SELECT的查询序列号。
select_type:表示SELECT语句的类型。
table:表示查询的表。
type:表示表的连接类型。
possible_keys:给出了MySQL在搜索数据记录时可选用的各个索引。
key:是MySQL实际选用的索引。
key_len:给出索引按字节计算的长度,key_len数值越小,表示越快。
ref:给出了关联关系中另一个数据表里的数据列名。
rows:是MySQL在执行这个查询时预计会从这个数据表里读出的数据行的个数。
Extra:提供了与关联操作有关的信息。
扩展阅读
DESCRIBE语句的使用方法与EXPLAIN语句是一样的,分析结果也是一样的,并且可以缩写成DESC。。
DESCRIBE语句的语法形式如下:
DESCRIBE SELECT select_options
4.5.7 explain关注什么?
参考答案:
重点要关注如下几列:
image-20220715163207459
其中,type包含以下几种结果,从上之下依次是最差到最好:
image-20220715163250325
另外,Extra列需要注意以下的几种情况:
image-20220715163326412
image-20220715163338331
4.6 其他
4.6.1 介绍一下数据库设计的三大范式
参考答案:
目前关系数据库有六种范式,一般来说,数据库只需满足第三范式(3NF)就行了。
第一范式(1NF):
是指在关系模型中,对于添加的一个规范要求,**所有的域都应该是原子性的,即数据库表的每一列都是
不可分割的原子数据项,而不能是集合,数组,记录等非原子数据项**。
即实体中的某个属性有多个值时,必须拆分为不同的属性。在符合第一范式表中的每个域值只能是实体
的一个属性或一个属性的一部分。简而言之,第一范式就是无重复的域
。
第二范式(2NF):
在1NF的基础上,非码 属性必须完全依赖于候选码(在1NF基础上消除 非主属性对主码的部分函数依
赖)。
第二范式是在第一范式的基础上建立起来的,即满足第二范式必须先满足第一范式。第二范式要求数据
库表中的每个实例或记录必须可以被唯一地区分。选取一个能区分每个实体的属性或属性组,作为实体
的唯一标识。
例如在员工表中的身份证号码即可实现每个一员工的区分,该身份证号码即为候选键,任何一个候选键
都可以被选作主键。在找不到候选键时,可额外增加属性以实现区分,如果在员工关系中,没有对其身
份证号进行存储,而姓名可能会在数据库运行的某个时间重复,无法区分出实体时,设计辟如ID等不重
复的编号以实现区分,被添加的编号或ID选作主键。
第三范式(3NF):
在2NF基础上,任何非主属性 不依赖于 其它非主属性(在2NF基础上消除传递依赖)。
第三范式是第二范式的一个子集,即满足第三范式必须满足第二范式。简而言之,第三范式要求一个关
系中不包含已在其它关系已包含的非主关键字信息。
例如,存在一个部门信息表,其中每个部门有部门编号(dept_id)、部门名称、部门简介等信息。那
么在员工信息表中列出部门编号后就不能再将部门名称、部门简介等与部门有关的信息再加入员工信息
表中。如果不存在部门信息表,则根据第三范式(3NF)也应该构建它,否则就会有大量的数据冗余。
4.6.2 说一说你对MySQL引擎的了解
参考答案
MySQL提供了多个不同的存储引擎,包括 处理事务安全表的引擎和 处理非事务安全表的引擎。在
MySQL中,不需要在整个服务器中使用同一种存储引擎,针对具体的要求,可以对每一个表使用不同的
存储引擎。MySQL 8.0支持的存储引擎有InnoDB、MyISAM、Memory、Merge、Archive、
Federated、CSV、BLACKHOLE等。其中,最常用的引擎是InnoDB和MyISAM。
InnoDB存储引擎:
InnoDB是事务型数据库的首选引擎,支持事务安全表(ACID),支持行锁定和外键。MySQL 5.5.5之
后,InnoDB作为默认存储引擎,主要特性如下:
InnoDB 给MySQL提供了具有提交、回滚和崩溃恢复能力的事务安全(ACID兼容)存储引擎。
InnoDB锁定在行级 并且也在SELECT语句中提供一个类似Oracle的非锁定读。这些功能增加了多用
户部署和性能。在SQL查询中,可以自由地将InnoDB类型的表与其他MySQL表的类型混合起来,
甚至在同一个查询中也可以混合。
InnoDB是为处理巨大数据量的最大性能设计。它的CPU效率可能是任何其他基于磁盘的关系数据
库引擎所不能匹敌的。
InnoDB存储引擎完全 与 MySQL服务器整合,为在主内存中缓存数据和索引而维持它自己的缓冲
池。InnoDB将它的表 和 索引存在一个逻辑表空间中,表空间可以包含数个文件(或原始磁盘分
区)。这与MyISAM表不同,比如在MyISAM表中每个表被存在分离的文件中。InnoDB表可以是
任何尺寸,即使在文件尺寸被限制为2GB的操作系统上。
InnoDB 支持 外键完整性约束(FOREIGN KEY)。存储表中的数据时,每张表的存储都按主键顺序
存放,如果没有显示在表定义时指定主键,InnoDB会为每一行生成一个6B的ROWID,并以此作
为主键。
InnoDB被用在众多需要高性能的大型数据库站点上。InnoDB不创建目录,使用InnoDB时,
MySQL将在数据目录下创建一个名为ibdata1的10MB大小的自动扩展数据文件,以及两个名为
ib_logfile0和ib_logfile1的5MB大小的日志文件。
MyISAM存储引擎:
MyISAM基于ISAM存储引擎,并对其进行扩展。它是在Web、数据仓储和其他应用环境下最常使用的存
储引擎之一。MyISAM拥有较高的插入、查询速度,但 不支持事务。MyISAM的主要特性如下:
在 支持大文件(达63位文件长度)的文件系统和操作系统上被支持。
当把 删除和更新及插入操作 混合使用的时候,动态尺寸的行产生更少碎片。这要通过合并相邻被删
除的块以及若下一个块被删除则扩展到下一块来自动完成。
每个MyISAM表 最大的索引数是64,这可以通过重新编译来改变。每个索引最大的列数是16个。
最大的键长度是1000B,这也可以通过编译来改变。对于键长度超过250B的情况,一个超过
1024B的键将被用上。
BLOB和TEXT列 可以被索引。
NULL值 被允许在索引的列中,这个值占每个键的0~1个字节。
所有数字键值 以高字节优先被存储,以允许一个更高的索引压缩。
每个表一个AUTO_INCREMENT列的内部处理。MyISAM为INSERT和UPDATE操作自动更新这一
列,这使得AUTO_INCREMENT列更快(至少10%)。在序列顶的值被删除之后就不能再利用。
可以把数据文件和索引文件放在不同目录。
每个字符列可以有不同的字符集。
有VARCHAR的表可以固定或动态记录长度。
VARCHAR和CHAR列可以多达64KB。
4.6.3 说一说你对redo log、undo log、binlog的了解
参考答案:
binlog(Binary Log):
二进制日志文件 就是常说的binlog。二进制日志记录了MySQL所有修改数据库的操作,然后以二进制的
形式记录在日志文件中,其中还包括每条语句所执行的时间和所消耗的资源,以及相关的事务信息。
默认情况下,二进制日志功能是开启的,启动时可以重新配置 --log-bin[=file_name] 选项,修改二进
制日志存放的目录和文件名称。
redo log:
重做日志用来实现事务的持久性,即事务ACID中的D。它由两部分组成:一是内存中的重做日志缓冲
(redo log buffer),其是易失的;二是重做日志文件(redo log file),它是持久的。
InnoDB是事务的存储引擎,它通过Force Log at Commit机制 实现事务的持久性, 即当事务提交
(COMMIT)时,必须先将该事务的所有日志写入到重做日志文件进行持久化,待事务的COMMIT操作
完成才算完成。这里的日志是指重做日志, 在InnoDB存储引擎中,由两部分组成,即redo log和undo
log。
redo log用来保证事务的持久性,undo log用来帮助事务回滚及MVCC的功能。
redo log基本上都是顺序写的,在数据库运行时不需要对redo log的文件进行读取操作。而undo log是需要进行随机读写的。
undo log:
重做日志记录了事务的行为,可以很好地通过其对页进行“重做”操作。但是事务有时还需要进行回滚操
作,这时就需要undo。因此在对数据库进行修改时,InnoDB存储引擎不但会产生redo,还会产生一定
量的undo。这样如果用户执行的事务或语句由于某种原因失败了,又或者用户用一条ROLLBACK语句
请求回滚,就可以利用这些undo信息将数据回滚到修改之前的样子。
redo存放在重做日志文件中,与redo不同,undo存放在数据库内部的一个特殊段(segment)中,这
个段称为undo段(undo segment),undo段位于共享表空间内。
4.6.4 谈谈你对MVCC的了解
参考答案
InnoDB默认的隔离级别是RR(REPEATABLE READ),RR解决脏读、不可重复读、幻读等问题,使用
的是MVCC。MVCC全称Multi-Version Concurrency Control,即多版本的并发控制协议。它最大的优
点是读不加锁,因此读写不冲突,并发性能好。InnoDB实现MVCC,多个版本的数据可以共存,主要基
于以下技术及数据结构:
隐藏**列: InnoDB中每行数据都有隐藏列**,隐藏列中包含了本行数据的事务id、指向undo log的指
针等。
基于undo log的版本链:每行数据的隐藏列中包含了指向undo log的指针,而每条undo log也会
指向更早版本的undo log,从而形成一条版本链。
ReadView:通过隐藏列和版本链,MySQL可以将数据恢复到指定版本。但是具体要恢复到哪个版
本,则需要根据ReadView来确定。所谓ReadView,是指事务(记做事务A)在某一时刻给整个事
务系统(trx_sys)打快照,之后再进行读操作时,会将读取到的数据中的事务id与trx_sys快照比
较,从而判断数据对该ReadView是否可见,即对事务A是否可见。
4.6.5 MySQL主从同步是如何实现的?
参考答案
复制(replication)是MySQL数据库提供的一种高可用高性能的解决方案,一般用来建立大型的应用。
总体来说,replication的工作原理分为以下3个步骤:
主服务器(master)把 数据更改记录到二进制日志(binlog)中。
从服务器(slave)把 主服务器的二进制日志复制到自己的中继日志(relay log)中。
从服务器 重做中继日志中的日志,把更改应用到自己的数据库上,以达到数据的最终一致性。
复制的工作原理并不复杂,其实就是一个完全备份加上二进制日志备份的还原。不同的是这个二进制日
志的还原操作基本上实时在进行中。这里特别需要注意的是,复制不是完全实时地进行同步,而是异步
实时。这中间存在主从服务器之间的执行延时,如果主服务器的压力很大,则可能导致主从服务器延时
较大。复制的工作原理如下图所示,其中从服务器有2个线程,一个是I/O线程,负责读取主服务器的二
进制日志,并将其保存为中继日志;另一个是SQL线程,复制执行中继日志。
image-20220715201610866
5. 设计模式
5.1 说说什么是单例设计模式,如何实现
参考回答
单例模式定义
保证一个类仅有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。
那么我们就必须保证:
(1)该类不能被复制。
(2)该类不能被公开的创造。
那么对于C++来说,它的构造函数,拷贝构造函数和赋值函数都不能被公开调用。
单例模式实现方式:
单例模式通常有两种模式,分别为懒汉式单例和饿汉式单例。两种模式实现方式分别如下:
(1)懒汉式设计模式实现方式(2种)
a. 静态指针 + 用到时初始化
b. 局部 静态变量
(2)饿汉式设计模式(2种)
a. 直接定义 静态对象
b. 静态指针 + 类外初始化时new空间实现
答案解析:
懒汉模式
懒汉模式的特点是延迟加载,比如配置文件,采用懒汉式的方法,配置文件的实例直到用到的
时候才会加载,不到万不得已就不会去实例化类,也就是说在第一次用到类实例的时候才会去实例
化。以下是懒汉模式实现方式C++代码:
(1)懒汉模式实现一:静态指针 + 用到时初始化
/代码实例(线程不安全)
template<typename T>
class Singleton
{
public:
static T& getInstance()
{
if (!value_)
{
value_ = new T();
}
return *value_;
}
private:
Singleton();
~Singleton();
static T* value_;
};
template<typename T>
T* Singleton<T>::value_ = NULL;
在 单线程中,这样的写法是可以正确使用的,但是在多线程中就不行了,该方法是线程不安全的。
a. 假如线程A和线程B, 这两个线程要访问getInstance函数,线程A进入getInstance函数,并检测
if条件,由于是第一次进入,value为空,if条件成立,准备创建对象实例。
b. 但是, 线程A有可能被OS的调度器中断而挂起睡眠,而将控制权交给线程B。
c. 线程B同样来到if条件,发现value还是为NULL,因为线程A还没来得及构造它就已经被中断
了。此时假设线程B完成了对象的创建,并顺利的返回。
d. 之后线程A被唤醒,继续执行new再次创建对象,这样一来,两个线程就构建两个对象实例,
这就破坏了唯一性。
另外,还存在内存泄漏的问题,new出来的东西始终没有释放,下面是一种饿汉式的一种改进。
//代码实例(线程安全)
Template<typename T>
class Singleton
{
public:
static T& getInstance()
{
if (!value_)
{
value_ = new T();
}
return *value_;
}
private:
class CGarbo
{
public:
~CGarbo()
{ if(Singleton::value_)
delete Singleton::value_;
}
};
static CGarbo Garbo;
Singleton();
~Singleton();
static T* value_;
};
template<typename T>
T* Singleton<T>::value_ = NULL;
在程序运行结束时,系统会 调用Singleton的静态成员Garbo的析构函数,该析构函数会删除单
例的唯一实例。使用这种方法释放单例对象有以下特征:
a. 在 单例类 内部定义 专有的嵌套类;
b. 在 单例类内 定义 私有的专门用于释放的静态成员;
c. 利用 程序 在结束时析构全局变量的特性,选择最终的释放时机。
(2)懒汉模式实现二:局部静态变量
/代码实例(线程不安全)
template<typename T>
class Singleton
{
public:
static T& getInstance()
{
static T instance;
return instance;
}
private:
Singleton(){};
Singleton(const Singleton&);
Singleton& operator=(const Singleton&);
};
同样,静态局部变量的实现方式也是线程不安全的。如果存在 多个单例对象的析构顺序有依赖
时,可能会出现程序崩溃的危险。
对于局部静态对象的也是一样的。因为 static T instance; 语句不是一个原子操作,在第一次被
调用时会调用Singleton的构造函数,而如果构造函数里如果有多条初始化语句,则初始化动作可
以分解为多步操作,就存在多线程竞争的问题。
为什么存在 多个单例对象的析构顺序有依赖时,可能会出现程序崩溃的危险?
原因 : 由于静态成员是在第一次调用函数GetInstance时进行初始化,调用 构造函数的,因此构
造函数的调用顺序时可以唯一确定了。对于析构函数,我们只知道其调用顺序和构造函数的调用顺
序相反,但是如果几个Singleton类的析构函数之间也有依赖关系,而且出现 类似单例实例A的析构
函数中使用了单例实例B,但是程序析构时是先调用实例B的析构函数,此时在A析构函数中使用B
时就可能会崩溃。
//代码实例(线程安全)
#include <string>
#include <iostream>
using namespace std;
class Log
{
public:
static Log* GetInstance()
{
static Log oLog;
return &oLog;
}
void Output(string strLog)
{
cout<<strLog<<(*m_pInt)<<endl;
}
private:
Log():m_pInt(new int(3))
{
}
~Log()
{cout<<"~Log"<<endl;
delete m_pInt;
m_pInt = NULL;
}
int* m_pInt;
};
class Context
{
public:
static Context* GetInstance()
{
static Context oContext;
return &oContext;
}
~Context()
{
Log::GetInstance()->Output(__FUNCTION__);
}
void fun()
{
Log::GetInstance()->Output(__FUNCTION__);
}
private:
Context(){}
Context(const Context& context);
};
int main(int argc, char* argv[])
{
Context::GetInstance()->fun();
return 0;
}
在这个反例中有两个Singleton: Log和Context,Context的fun和析构函数会调用Log来输出一
些信息, 结果程序Crash掉了,该程序的运行的序列图如下(其中画红框的部分是出问题的部
分):
image-20220715204652431
解决方案:对于析构的顺序,我们可以用一个容器来管理它,根据单例之间的依赖关系释放实
例,对所有的实例的析构顺序进行排序,之后调用各个单例实例的析构方法,如果出现了循环依赖
关系,就给出异常,并输出循环依赖环。
饿汉模式
单例类定义的时候就进行实例化。 因为main函数执行之前, 全局作用域 的类成员静态变量
m_Instance已经初始化,故没有多线程的问题。
(1)饿汉模式实现一:直接定义静态对象
//代码实例(线程安全)
//.h文件
class Singleton
{
public:
static Singleton& GetInstance();
private:
Singleton(){}
Singleton(const Singleton&);
Singleton& operator= (const Singleton&);
private:
static Singleton m_Instance;
};
//CPP文件
Singleton Singleton::m_Instance;//类外定义-不要忘记写
Singleton& Singleton::GetInstance()
{
return m_Instance;
}
//函数调用
Singleton& instance = Singleton::GetInstance();
优点:
实现简单,多线程安全。
缺点:
a. 如果存在多个单例对象且这几个单例对象相互依赖,可能会出现程序崩溃的危险。原因:对
编译器来说,静态成员变量的初始化顺序和析构顺序是一个未定义的行为;具体分析在懒汉模式中
也讲到了。
b. 在程序开始时,就创建类的实例,如果Singleton对象产生很昂贵,而本身有很少使用,这
种方式单从资源利用效率的角度来讲,比懒汉式单例类稍差些。但从反应时间角度来讲,则比懒汉
式单例类稍好些。
使用条件:
a. 当肯定 不会有构造和析构依赖关系的情况。
b. 想避免 频繁加锁时的性能消耗
//代码实例(线程安全)
class Singleton
{
protected:
Singleton(){}
private:
static Singleton* p;
public:
static Singleton* initance();
};
Singleton* Singleton::p = new Singleton;
Singleton* singleton::initance()
{
return p;
}
5.2 简述一下单例设计模式的懒汉式和饿汉式,如何保证线程安全
参考回答:
懒汉式设计模式
懒汉模式的特点是延迟加载,比如配置文件,采用懒汉式的方法,配置文件的实例直到用到的
时候才会加载,不到万不得已就不会去实例化类,也就是说在第一次用到类实例的时候才会去实例
化。
饿汉模式
单例类定义 的时候就进行实例化。因为main函数执行之前,全局作用域 的类成员静态变量
m_Instance已经初始化,故没有多线程的问题。
答案解析:
懒汉设计模式两种实现方式线程不安全问题的解决:
(1)懒汉模式实现一:静态指针 + 用到时初始化
//代码实例(线程不安全)
template<typename T>
class Singleton{
public:
static T& getInstance()
{
if (!value_)
{
value_ = new T();
}
return *value_;
}
private:
Singleton();
~Singleton();
static T* value_;
};
template<typename T>
T* Singleton<T>::value_ = NULL;
在单线程中,这样的写法是可以正确使用的,但是在多线程中就不行了,该方法是线程不安全的。
a. 假如线程A和线程B, 这两个线程要访问getInstance函数,线程A进入getInstance函数,并检测if条
件,由于是第一次进入,value为空,if条件成立,准备创建对象实例。
b. 但是,线程A有可能被OS的调度器中断而挂起睡眠,而将控制权交给线程B。
c. 线程B同样来到if条件,发现value还是为NULL,因为线程A还没来得及构造它就已经被中断了。此
时假设线程B完成了对象的创建,并顺利的返回。
d. 之后线程A被唤醒,继续执行new再次创建对象,这样一来,两个线程就构建两个对象实例,这就破
坏了唯一性。
另外,还存在内存泄漏的问题,new出来的东西始终没有释放,下面是一种饿汉式的一种线程安全的改
进。
//代码实例(线程安全)
Template<typename T>
class Singleton
{
public:
static T& getInstance()
{
if (!value_)
{
value_ = new T();
}
return *value_;
}
private:
class CGarbo
{
public:
~CGarbo()
{
if(Singleton::value_)
delete Singleton::value_;
}
};
static CGarbo Garbo;
Singleton();
~Singleton();
static T* value_;};
template<typename T>
T* Singleton<T>::value_ = NULL;
在程序运行结束时,系统会调用Singleton的静态成员Garbo的析构函数,该析构函数会删除单例的唯
一实例。使用这种方法释放单例对象有以下特征:
a. 在单例类内部定义专有的嵌套类;
b. 在单例类内定义私有的专门用于释放的静态成员;
c. 利用程序在结束时析构全局变量的特性,选择最终的释放时机。
(2)懒汉模式实现二:局部静态变量
//代码实例(线程不安全)
template<typename T>
class Singleton
{
public:
static T& getInstance()
{
static T instance;
return instance;
}
private:
Singleton(){};
Singleton(const Singleton&);
Singleton& operator=(const Singleton&);
};
同样,静态局部变量的实现方式也是线程不安全的。如果存在多个单例对象的析构顺序有依赖时,可
能会出现程序崩溃的危险。
对于局部静态对象的也是一样的。因为 static T instance;语句不是一个原子操作,在第一次被调用
时会调用Singleton的构造函数,而如果构造函数里如果有多条初始化语句,则初始化动作可以分解为
多步操作,就存在多线程竞争的问题。
为什么存在多个单例对象的析构顺序有依赖时,可能会出现程序崩溃的危险?
原因:由于静态成员是在第一次调用函数GetInstance时进行初始化,调用构造函数的,因此构造函
数的调用顺序时可以唯一确定了。对于析构函数,我们只知道其调用顺序和构造函数的调用顺序相反,
但是如果几个Singleton类的析构函数之间也有依赖关系,而且出现类似单例实例A的析构函数中使用了
单例实例B,但是程序析构时是先调用实例B的析构函数,此时在A析构函数中使用B时就可能会崩溃。
//代码实例(线程安全)
#include <string>
#include <iostream>
using namespace std;
class Log
{
public:
static Log* GetInstance()
{
static Log oLog;
return &oLog;
}
void Output(string strLog)
{
cout<<strLog<<(*m_pInt)<<endl;
}
private:
Log():m_pInt(new int(3))
{
}
~Log()
{cout<<"~Log"<<endl;
delete m_pInt;
m_pInt = NULL;
}
int* m_pInt;
};
class Context
{
public:
static Context* GetInstance()
{
static Context oContext;
return &oContext;
}
~Context()
{
Log::GetInstance()->Output(__FUNCTION__);
}
void fun()
{
Log::GetInstance()->Output(__FUNCTION__);
}
private:
Context(){}
Context(const Context& context);
};
int main(int argc, char* argv[])
{
Context::GetInstance()->fun();
return 0;
}
在这个反例中有两个Singleton: Log和Context,Context的fun和析构函数会调用Log来输出一些信
息,结果程序Crash掉了,该程序的运行的序列图如下(其中画红框的部分是出问题的部分):
image-20220715211552147
‘解决方案:对于析构的顺序,我们可以用一个容器来管理它,根据单例之间的依赖关系释放实例,对
所有的实例的析构顺序进行排序,之后调用各个单例实例的析构方法,如果出现了循环依赖关系,就给
出异常,并输出循环依赖环。
5.3 请说说工厂设计模式,如何实现,以及它的优点
参考回答
工厂设计模式的定义
定义一个创建对象的接口,让 子类 决定 实例化哪个类,而对象的创建 统一交由工厂去生产,有
良好的封装性,既做到了解耦,也保证了最少知识原则。
工厂设计模式分类
工厂模式属于创建型模式,大致可以分为三类,简单工厂模式、工厂方法模式、抽象工厂模
式。听上去差不多,都是工厂模式。下面一个个介绍:
(1)简单工厂模式
它的主要特点是需要 在工厂类中做判断,从而创造相应的产品。当增加新的产品时,就需要修
改工厂类。
举例:有一家生产 处理器核的厂家,它只有一个工厂,能够生产两种型号的处理器核。客户 需
要什么样的处理器核,一定要显示地告诉生产工厂。下面给出一种实现方案:
//程序实例(简单工厂模式)
enum CTYPE {COREA, COREB};
class SingleCore
{
public:
virtual void Show() = 0;
}; //单核A
class SingleCoreA: public SingleCore
{
public:
void Show() { cout<<"SingleCore A"<<endl; }
};
//单核B
class SingleCoreB: public SingleCore
{
public:
void Show() { cout<<"SingleCore B"<<endl; }
};
//唯一的工厂,可以生产两种型号的处理器核,在内部判断
class Factory
{
public:
SingleCore* CreateSingleCore(enum CTYPE ctype)
{
if(ctype == COREA) //工厂内部判断
return new SingleCoreA(); //生产核A
else if(ctype == COREB)
return new SingleCoreB(); //生产核B
else
return NULL;
}
};
优点: 简单工厂模式可以根据需求,动态生成使用者所需类的对象,而使用者 不用 去知道怎么
创建对象,使得各个模块各司其职,降低了系统的耦合性。
缺点:就是要增加 新的核类型时,就需要修改工厂类。这就违反了开放封闭原则:软件实体
(类、模块、函数)可以扩展,但是不可修改。
(2)工厂方法模式
所谓 工厂方法模式,是指 定义 一个用于创建对象的接口,让子类决定实例化哪一个类。Factory
Method 使一个类的实例化延迟到其子类。
举例: 这家生产处理器核的产家赚了不少钱, 于是决定再开设一个工厂**专门用来生产B型号的单
核,而原来的工厂专门用来生产A型号的单核**。这时,客户要做的是找好工厂,比如要A型号的
核, 就找A工厂要;否则找B工厂要, 不再需要告诉工厂具体要什么型号的处理器核了。下面给出
一个实现方案:
/程序实例(工厂方法模式)
class SingleCore
{
public:
virtual void Show() = 0;
};
//单核A
class SingleCoreA: public SingleCore
{
public:
void Show() { cout<<"SingleCore A"<<endl; }
};
//单核B
class SingleCoreB: public SingleCore
{ public:
void Show() { cout<<"SingleCore B"<<endl; }
};
class Factory
{
public:
virtual SingleCore* CreateSingleCore() = 0;
};
//生产A核的工厂
class FactoryA: public Factory
{
public:
SingleCoreA* CreateSingleCore() { return new SingleCoreA; }
};
//生产B核的工厂
class FactoryB: public Factory
{
public:
SingleCoreB* CreateSingleCore() { return new SingleCoreB; }
};
优点: 扩展性好,符合了开闭原则,新增一种产品时,只需增加改对应的产品类和对应的工厂
子类即可。
缺点:每增加一种产品,就需要增加一个对象的工厂。如果这家公司发展迅速,推出了很多新
的处理器核,那么就要开设相应的新工厂。在C++实现中,就是要定义一个个的工厂类。显然,相
比简单工厂模式,工厂方法模式需要更多的类定义。
(3)抽象工厂模式
举例:这家公司的技术不断进步,不仅可以生产单核处理器,也能生产多核处理器。现在简单
工厂模式和工厂方法模式都鞭长莫及。抽象工厂模式登场了。它的定义 为 提供一个创建一系列相关
或相互依赖对象的接口,而无需指定它们具体的类。具体这样应用,这家公司还是开设两个工厂,
一个专门用来生产A型号的单核 多核处理器, 而另一个工厂专门用来生产B型号的单核多核处理
器,下面给出实现的代码:
/程序实例(抽象工厂模式)
//单核
class SingleCore
{
public:
virtual void Show() = 0;
};
class SingleCoreA: public SingleCore
{
public:
void Show() { cout<<"Single Core A"<<endl; }
};
class SingleCoreB :public SingleCore
{
public:
void Show() { cout<<"Single Core B"<<endl; }
};
//多核
class MultiCore
{
public:
virtual void Show() = 0; };
class MultiCoreA : public MultiCore
{
public:
void Show() { cout<<"Multi Core A"<<endl; }
};
class MultiCoreB : public MultiCore
{
public:
void Show() { cout<<"Multi Core B"<<endl; }
};
//工厂
class CoreFactory
{
public:
virtual SingleCore* CreateSingleCore() = 0;
virtual MultiCore* CreateMultiCore() = 0;
};
//工厂A,专门用来生产A型号的处理器
class FactoryA :public CoreFactory
{
public:
SingleCore* CreateSingleCore() { return new SingleCoreA(); }
MultiCore* CreateMultiCore() { return new MultiCoreA(); }
};
//工厂B,专门用来生产B型号的处理器
class FactoryB : public CoreFactory
{
public:
SingleCore* CreateSingleCore() { return new SingleCoreB(); }
MultiCore* CreateMultiCore() { return new MultiCoreB(); }
};
优点: 工厂抽象类创建了多个类型的产品,当有需求时,可以创建相关产品子类和子工厂类来
获取。
缺点: 扩展新种类产品时困难。抽象工厂模式 需要我们在工厂抽象类中提前确定了可能需要的
产品种类,以满足不同型号的多种产品的需求。但是 如果我们需要的产品种类并没有在工厂抽象类
中提前确定,那我们就需要去修改工厂抽象类了,而一旦修改了工厂抽象类,那么所有的工厂子类
也需要修改,这样显然扩展不方便。
答案解析
三种工厂模式的UML图如下:
简单工厂模式UML
image-20220715220627984
工厂方法的UML图
image-20220715220800542
抽象工厂模式的UML图
image-20220715220813074
5.4 请说说装饰器计模式,以及它的优缺点
参考回答:
装饰器计模式的定义
指在不改变现有对象结构的情况下,动态地给该对象增加一些职责(即增加其额外功能)的模
式,它属于对象结构型模式。
优点
(1)装饰器 是继承的有力补充,比 继承灵活,在不改变原有对象的情况下,动态的给一个对象扩
展功能,即插即用;
(2)通过使用 不用装饰类及这些装饰类的排列组合,可以实现不同效果;
(3)装饰器模式 完全遵守开闭原则。
缺点
装饰模式 会增加 许多子类,过度使用会增加程序得复杂性。
装饰模式的结构与实现
通常情况下,扩展一个类的功能会使用继承方式来实现。但继承具有静态特征,耦合度高,并
且随着扩展功能的增多,子类会很膨胀。如果使用组合关系来创建一个包装对象(即装饰对象)来
包裹真实对象,并在保持真实对象的类结构不变的前提下,为其提供额外的功能,这就是装饰模式
的目标。下面来分析其基本结构和实现方法。
装饰模式主要包含以下角色:
(1)抽象 构件(Component)角色: 定义一个抽象接口以规范准备接收附加责任的对象。
(2)具体 构件(ConcreteComponent)角色: 实现抽象构件,通过装饰角色为其添加一些职
责。
(3)抽象装饰(Decorator)角色: 继承抽象构件,并包含具体构件的实例,可以通过其子类扩
展具体构件的功能。
(4)具体装饰(ConcreteDecorator)角色: 实现抽象装饰的相关方法,并给具体构件对象添加
附加的责任。装饰模式的结构图如下图所示:
image-20220715221734744
装饰模式的实现代码如下:
#include <string>
#include <iostream>
//基础组件接口定义了可以被装饰器修改的操作
class Component {
public:
virtual ~Component() {}
virtual std::string Operation() const = 0;
};
//具体组件提供了操作的默认实现。这些类在程序中可能会有几个变体
class ConcreteComponent : public Component {
public:
std::string Operation() const override {
return "ConcreteComponent";
}
};
//装饰器基类和其他组件遵循相同的接口。这个类的主要目的是为所有的具体装饰器定义封装接口。
//封装的默认实现代码中可能会包含一个保存被封装组件的成员变量,并且负责对齐进行初始化
class Decorator : public Component {
protected:
Component* component_;
public:
Decorator(Component* component) : component_(component) {
}
//装饰器会将所有的工作分派给被封装的组件 std::string Operation() const override {
return this->component_->Operation();
}
};
//具体装饰器必须在被封装对象上调用方法,不过也可以自行在结果中添加一些内容。
class ConcreteDecoratorA : public Decorator {
//装饰器可以调用父类的是实现,来替代直接调用组件方法。
public:
ConcreteDecoratorA(Component* component) : Decorator(component) {
}
std::string Operation() const override {
return "ConcreteDecoratorA(" + Decorator::Operation() + ")";
}
};
//装饰器可以在调用封装的组件对象的方法前后执行自己的方法
class ConcreteDecoratorB : public Decorator {
public:
ConcreteDecoratorB(Component* component) : Decorator(component) {
}
std::string Operation() const override {
return "ConcreteDecoratorB(" + Decorator::Operation() + ")";
}
};
//客户端代码可以使用组件接口来操作所有的具体对象。这种方式可以使客户端和具体的实现类脱耦
void ClientCode(Component* component) {
// ...
std::cout << "RESULT: " << component->Operation();
// ...
}
int main() {
Component* simple = new ConcreteComponent;
std::cout << "Client: I've got a simple component:\n";
ClientCode(simple);
std::cout << "\n\n";
Component* decorator1 = new ConcreteDecoratorA(simple);
Component* decorator2 = new ConcreteDecoratorB(decorator1);
std::cout << "Client: Now I've got a decorated component:\n";
ClientCode(decorator2);
std::cout << "\n";
delete simple;
delete decorator1;
delete decorator2;
return 0;
}
5.5 请说说观察者设计模式,如何实现
参考回答
观察者设计模式的定义
多个对象间存在 一对多的依赖关系,当 一个对象的状态发生改变时,所有依赖于它的对象都
得到通知并被自动更新。这种模式有时又称作 发布-订阅模式、模型-视图模式,它是对象行为型模
式。
优点
(1)降低了 目标与 观察者之间的耦合关系,两者之间是抽象耦合关系。符合 依赖倒置 原 则。
(2)目标与观察者之间建立了一套触发机制。
缺点
(1)目标与观察者之间的依赖关系并没有完全解除,而且有可能出现循环引用。
(2)当观察者对象很多时,通知的发布会花费很多时间,影响程序的效率。
观察者设计模式的结构与实现
观察者模式的主要角色如下
(1)抽象主题(Subject)角色:也叫 抽象目标类,它提供了一个用于保存观察者对象的聚集类和
增加、删除观察者对象的方法,以及通知所有观察者的抽象方法。
(2)具体主题(Concrete Subject)角色:也叫 具体目标类,它实现抽象目标中的通知方法,当
具体主题的内部状态发生改变时,通知所有注册过的观察者对象。
(3)抽象观察者(Observer)角色:它是一个抽象类或接口,它包含了一个更新自己的抽象方
法,当接到具体主题的更改通知时被调用。
(4)具体观察者(Concrete Observer)角色: 实现 抽象观察者中定义的抽象方法,以便在得到
目标的更改通知时更新自身的状态。
可以举个博客订阅的例子,当博主发表新文章的时候,即博主状态发生了改变,那些订阅的读者就会
收到通知,然后进行相应的动作, 比如去看文章,或者收藏起来。博主与读者之间存在种一对多的依赖
关系。下面给出相应的UML图设计:
image-20220715223048100
可以看到博客类中有一个观察者链表(即订阅者),当 博客的状态发生变化时,通过Notify成员函数
通知所有的观察者,告诉他们博客的状态更新了。而观察者通过Update成员函数获取博客的状态信
息。代码实现不难,下面给出C++的一种实现。
//观察者
class Observer
{
public:
Observer() {}
virtual ~Observer() {}
virtual void Update() {}
};
//博客
class Blog
{
public:
Blog() {}
virtual ~Blog() {}
void Attach(Observer *observer) { m_observers.push_back(observer); }
//添加观察者
void Remove(Observer *observer) { m_observers.remove(observer); } //移除观察者
void Notify() //通知观察者
{
list<Observer*>::iterator iter = m_observers.begin();
for(; iter != m_observers.end(); iter++)
(*iter)->Update();
}
virtual void SetStatus(string s) { m_status = s; } //设置状态
virtual string GetStatus() { return m_status; } //获得状态
private:
list<Observer* > m_observers; //观察者链表
protected:
string m_status; //状态
};
以上是观察者和博客的基类,定义了通用接口。博客类 主要完成 观察者的添加、移除、通知操作,设
置和获得状态仅仅是一个默认实现。下面给出它们相应的子类实现。
/具体博客类
class BlogCSDN : public Blog
{
private:
string m_name; //博主名称
public:
BlogCSDN(string name): m_name(name) {}
~BlogCSDN() {}
void SetStatus(string s) { m_status = "CSDN通知 : " + m_name + s; } //具体设置状态信息
string GetStatus() { return m_status; }
};
//具体观察者
class ObserverBlog : public Observer
{
private:
string m_name; //观察者名称Blog *m_blog; //观察的博客,当然以链表形式更好,就可以观察多个博客
public:
ObserverBlog(string name,Blog *blog): m_name(name), m_blog(blog) {}
~ObserverBlog() {}
void Update() //获得更新状态
{
string status = m_blog->GetStatus();
cout<<m_name<<"-------"<<status<<endl;
}
};
//测试案例
int main()
{
Blog *blog = new BlogCSDN("wuzhekai1985");
Observer *observer1 = new ObserverBlog("tutupig", blog);
blog->Attach(observer1);
blog->SetStatus("发表设计模式C++实现(15)——观察者模式");
blog->Notify();
delete blog;
delete observer1;
return 0;
}
6.数据结构与算法
讲一讲,如何构造一致性哈希算法。
考察点:哈希算法
参考回答:
先构造一个长度为2^32的整数环(这个环被称为一致性Hash环),根据节点名称的Hash值(其分布为
[0, 2^32-1]) 将服务器节点放置在这个Hash环上,然后根据 数据的Key值计算得到其Hash值(其分布
也为[0, 2^32-1]),接着在Hash环上顺时针查找距离这个Key值的Hash值最近的服务器节点,完成Key
到服务器的映射查找。
这种算法解决了普通余数Hash算法伸缩性差的问题,可以保证在上线、下线服务器的情况下尽量有多
的请求命中原来路由到的服务器。
’
如何打印二叉树每层的节点?
考察点:二叉树
注意:面试中一般写核心代码,另外就是可以问一下面试是输出ArrayList还是int数组。
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode(int x) { val = x; }
* }
*/
class Solution {
public int[] levelOrder(TreeNode root) {
// 如果root为空直接返回
if(root == null){
return new int[0];
}
// 层次遍历
// 队列
Deque<TreeNode> deque = new LinkedList<>();
// 存每一层节点
ArrayList<Integer> temp = new ArrayList<>(); // 根节点入队
deque.offer(root);
while(!deque.isEmpty()){
int size = deque.size();
for(int i=0;i<size;i++) {
//层次遍历
TreeNode curroot = deque.pop();
temp.add(curroot.val);
// 做节点是否为空
if(curroot.left!=null)
deque.offer(curroot.left);
if(curroot.right!=null)
deque.offer(curroot.right);
}
}
// 如果要求返回int数组 就写上这一步,如果没有要求可以直接返回temp即可。
int []result = new int[temp.size()];
for(int i=0;i<result.length;i++){
result[i] = temp.get(i);
}
return result;
}
}
如何知道二叉树的深度?
考察点:二叉树
参考回答:
求二叉树的深度方式有两种,递归以及非递归。(出自剑指offer)
1递归实现:
为了求树的深度,可以先求其左子树的深度和右子树的深度,可以用递归实现,递归的出口就是节点为
空。返回值为0;
/**
public class TreeNode {
int val = 0;
TreeNode left = null;
TreeNode right = null;
public TreeNode(int val) {
this.val = val;
}
}
*/
public class Solution {
public int TreeDepth(TreeNode root) {
if(root==null) return 0;
int left = TreeDepth(root.left);
int right = TreeDepth(root.right);
return (left>right)?(left+1):(right+1);
}
}
2非递归实现:
利用层次遍历的算法,将每一层的所有节点放入队列中,在将每一层节点的左右子节点存入放入到队列
中,用一个变量记录树的高度即可。
import java.util.*;
public class Solution {
public int TreeDepth(TreeNode root) {
if(root==null)
return 0;
Queue<TreeNode> result = new LinkedList<>();
result.add(root);
int height = 0;
while(!result.isEmpty()){
//获取当前的根节点
int size = result.size();
while(size>0){//遍历当前层,将该层的子节点都加入队列中
TreeNode nowroot = result.poll();
if(nowroot.left!=null)
result.add(nowroot.left);
if(nowroot.right!=null)
result.add(nowroot.right);
size = size-1;//
}
height += 1;//高度加1
}
return height;
}
}
二叉树任意两个节点之间路径的最大长度
int maxDist(Tree root) {
//如果树是空的,则返回0
if(root == NULL)
return 0;
if(root->left != NULL) {
root->lm = maxDist(root->left) +1;
}
if(root->right != NULL)
root->rm = maxDist(root->right) +1;
//如果以该节点为根的子树中有最大的距离,那就更新最大距离
int sum = root->rm + root->lm;
if(sum > max) {
max = sum;
}
return root->rm > root->lm ?root->rm : root->lm;
}
算法题:二叉树层序遍历,进一步提问:要求每层打印出
一个换行符
public
List<List<Integer>> levelOrder(TreeNode root) {
// 存最终结果
List<List<Integer>> result = new ArrayList<List<Integer>>();
// 队列
Queue<TreeNode> queue = new LinkedList<TreeNode>();
if (root == null) {
return result;
}
queue.offer(root);
// 层次遍历
while (queue.size() != 0) {
List<Integer> temp = new ArrayList<Integer>();
int size = queue.size();
for (int i = 0; i < size; i++) {
TreeNode temp = queue.poll();
temp.add(temp.val);
// 左右子树是否为空
if (temp.left != null) {
queue.offer(temp.left);
}
if (temp.right != null) {
queue.offer(temp.right);
}
}
result.add(temp);
}
return result;
}
怎么求一个二叉树的深度?手撕代码?
类似上面求二叉树的深度的题,这里给出递归的方式。
public class Solution {
public int TreeDepth(TreeNode root) {
if(root==null)
return 0;
int left = TreeDepth(root.left);
int right = TreeDepth(root.right);
return (left>right)?(left+1):(right+1);
}
}
请你说一下,B+树和B-树?
参考回答:
B+树内节点不存储数据,所有 data 存储在叶节点导致查询时间复杂度固定为 log n。而B-树查询
时间复杂度不固定,与 key 在树中的位置有关,最好为O(1)。
B+树叶节点 两两相连可大大增加区间访问性,可使用在范围查询等,而B-树每个节点 key 和 data
在一起,则无法区间查找。
B+树 更适合 外部存储。由于内节点无 data 域,每个节点能索引的范围更大更精确
6.3遍历
二叉树 Z 字型遍历
import java.util.*;
public class Solution {
public ArrayList<ArrayList<Integer>> Print(TreeNode pRoot) {
ArrayList<ArrayList<Integer>> result = new ArrayList<>();
Queue<TreeNode> queue = new LinkedList<TreeNode>();
//标记奇偶层 开始为0层 从左->右遍历
boolean flag = true;
if (pRoot == null) {
return result;
}
queue.add(pRoot);
while (queue.size() != 0) {
flag = !flag;
int size = queue.size();
ArrayList<Integer> temp = new ArrayList<Integer>();
for (int i = 0; i < size; i++) {
TreeNode curroot = queue.poll();
temp.add(curroot.val);
if (curroot.left != null)
queue.add(curroot.left);
if (curroot.right != null)
queue.add(curroot.right);
}
// 反转
if (flag) {
Collections.reverse(temp);
}
result.add(temp);
}
return result;
}
}
编程题:写一个函数,找到一个文件夹下所有文件,包括子文件夹
考察点:遍历
import java.io.File;public class Counter2 {
public static void main(String[] args) {
//取得目标目录
File file = new File("D:");
//获取目录下子文件及子文件夹
File[] files = file.listFiles();
readfile(files);
}
public static void readfile(File[] files) {
if(files == null) {// 如果目录为空,直接退出
return;
}
for(File f:files) {
//如果是文件,直接输出名字
if(f.isFile()) {
System.out.println(f.getName());
}
//如果是文件夹,递归调用
else if(f.isDirectory()) {
readfile(f.listFiles());
}
}
}
}
6.4链表
现在有一个单向链表,谈一谈,如何判断链表中是否出现
了环
考察点:链表
单链表有环,是指单链表中某个节点的next指针域指向的是链表中在它之前的某一个节点,这样在链表
的尾部形成一个环形结构。
解法:定义两个指针,同时从链表的头节点出发,一个指针一次走一步,另一个指针一次走两步。如果
走得快的指针追上了走得慢的指针,那么链表就是环形链表; 如果走得快的指针走到了链表的末尾
(next指向 NULL)都没有追上第一个指针,那么链表就不是环形链表。
谈一谈,bucket如果用链表存储,它的缺点是什么?
考察点:链表
参考回答:
1查找速度慢,因为查找时,需要循环链表访问
2如果进行频繁插入和删除操作,会导致速度很慢。
有一个链表,奇数位升序偶数位降序,如何将链表变成升序
public class
OddIncreaseEvenDecrease {
/**
* 按照奇偶位拆分成两个链表
* @param head
* @return
*/
// 例子 1 6 3 4 5 2 变成 1 2 3 4 5 6
public static Node[] getLists(Node head){
Node head1 = null;
Node head2 = null;
Node cur1 = null;
Node cur2 = null;
int count = 1;//用来计数
while(head != null){
// 遇到1 3 5
if(count % 2 == 1){
if(cur1 != null){
cur1.next = head;
cur1 = cur1.next;
}else{
cur1 = head;
head1 = cur1;
}
}else{
if(cur2 != null){
cur2.next = head;
cur2 = cur2.next;
}else{
cur2 = head;
head2 = cur2;
}
}
head = head.next;
count++;
}
//跳出循环,要让最后两个末尾元素的下一个都指向null
cur1.next = null;
cur2.next = null;
Node[] nodes = new Node[]{head1,head2};
return nodes;
}
/**
* 反转链表
* @param head * @return
*/
public static Node reverseList(Node head){
Node pre = null;
Node next = null;
while(head != null){
next = head.next;
head.next = pre;
pre = head;
head = next;
}
return pre;
}
/**
* 合并两个有序链表
* @param head1
* @param head2
* @return
*/
public static Node CombineList(Node head1,Node head2){
if(head1 == null || head2 == null){
return head1 != null ? head1 :head2;
}
Node head = head1.value < head2.value ?head1 : head2;
Node cur1 = head == head1 ? head1 :head2;
Node cur2 = head == head1 ? head2 :head1;
Node pre = null;
Node next = null;
while(cur1 != null && cur2 !=null){
if(cur1.value <= cur2.value){//这里一定要有=,否则一旦cur1的value和cur2的
value相等的话,下面的pre.next会出现空指针异常
pre = cur1;
cur1 = cur1.next;
}else{
next = cur2.next;
pre.next = cur2;
cur2.next = cur1;
pre = cur2;
cur2 = next;
}
}
pre.next = cur1 == null ? cur2 : cur1;
return head;
}
}
随机链表的复制
public RandomListNode copyRandomList(RandomListNode head) {
if (head == null)
return null;
RandomListNode p = head;
// copy every node and insert to list
while (p != null) {
RandomListNode copy = new RandomListNode(p.label);
copy.next = p.next;
p.next = copy;
p = copy.next;
}
// copy random pointer for each new node
p = head;
while (p != null) {
if (p.random != null)
p.next.random = p.random.next;
p = p.next.next;
}
// break list to two
p = head;
RandomListNode newHead = head.next;
while (p != null) {
RandomListNode temp = p.next;
p.next = temp.next;
if (temp.next != null)
temp.next = temp.next.next;
p = p.next;
}
return newHead;
}
如何反转单链表
public static Node reverseList(ListNode head){
ListNode pre = null;
ListNode next = null;
while(head != null){
next = head.next;
head.next = pre;
pre = head;
head = next;
}
return pre;
}
写一个算法,可以将一个二维数组顺时针旋转90度,说一下思路。
考察点:数组
public void rotate(int[][] matrix) {
int n = matrix.length;
for (int i = 0; i < n/2; i++) {
for (int j = i; j < n-1-i; j++)
{
int temp = matrix[i][j];
matrix[i][j] = matrix[n-1-j][i];
matrix[n-1-j][i] = matrix[n-1-i][n-1-j];
matrix[n-1-i][n-1-j] = matrix[j][n-1-i];
matrix[j][n-1-i] = temp;
}
}
}
一个数组,除一个元素外其它都是两两相等,求那个元素?
考察点:数组
解法:位运算,数组中的全部元素的异或运算结果即为数组中只出现一次的数字。
public static int find1From2(int[] a){
int len = a.length, result = 0;
for(int i = 0; i < len; i++){
result = result ^ a[i];
}
return result;
}
找出数组中和为S的一对组合,找出一组就行
考察点:数组
解法:两数之和经典题,找到一组即可返回,使用HashMap即可。
public int[] twoSum(int[] nums, int target) {
HashMap<Integer, Integer> map = new HashMap<Integer, Integer>();
int[] a = new int[2];
map.put(nums[0], 0);
for (int i = 1; i < nums.length;i++) {
if (map.containsKey(target - nums[i])) {
a[0] = map.get(target - nums[i]);
a[1] = i;
return a;
} else {
map.put(nums[i], i);
}
} return a;
}
求一个数组中连续子向量的最大和
考察点:数组
public int maxSubArray(int[] nums) {
int sum = 0;
int maxSum = Integer.MIN_VALUE;
if (nums == null || nums.length == 0) {
return sum;
}
for (int i = 0; i < nums.length;i++) {
sum += nums[i];
maxSum = Math.max(maxSum, sum);
if (sum < 0) {
sum = 0;
}
}
return maxSum;
}
寻找一数组中前K个最大的数
考察点:数组
解法:考场排序,可以用堆排序,也可以用快排,面试的时候看面试官怎么要求。这里给出快排的解
法,自己也可以尝试其他的方法。
public int findKthLargest(int[] nums, int k) {
if (k < 1 || nums == null) {
return 0;
}
return getKth(nums.length - k +1, nums, 0,nums.length - 1);
}
// 快排模板
public int getKth(int k, int[] nums, int start, int end) {
int pivot = nums[end];
int left = start;
int right = end;
while (true) {
while(nums[left] < pivot && left < right) {
left++;
}
while(nums[right] >= pivot && right > left) {
right--;
}
if(left == right) {
break;
}
swap(nums,left, right);
}
swap(nums, left, end);
if (k == left + 1) {
return pivot;
} else if (k < left + 1) {
return getKth(k, nums, start, left - 1);
} else {
return getKth(k, nums, left + 1, end);
}
}
// 交换元素值
public void swap(int[] nums, int n1, int n2) {
int tmp = nums[n1];
nums[n1] = nums[n2];
nums[n2] = tmp;
}
6.6排序
用java写一个冒泡排序?
public static void main(String[] args) {
int[] result = {2,4,1,3,6,5};
int temp;
System.out.println("----冒泡排序前顺序----");
for (int i : result) {
System.out.print(i);
}
for(int i=0;i<result.length-1;i++){
for(int j = 0;j<result.length-1-i;j++){
if(result[j+1]<result[j]){
//后一个比前一个小
temp = result[j];
result[j] = result[j+1];
result[j+1] = temp;
}
}
}
System.out.println();
System.out.println("----冒泡排序后结果----");
for (int i : result) {
System.out.print(i);
}
}
介绍一下,排序都有哪几种方法?请列举出来。
考察点:排序
参考回答:
排序的方法有:
插入排序(简单插入排序、希尔排序)
交换排序(冒泡排序、快速排序)
选择排序(简单选择排序、堆排序)
归并排序
分配排序(箱排序、基数排序)
绍一下,归并排序的原理是什么?
参考回答:
(1)归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and
Conquer)的一个非常典型的应用。
(2)首先考虑下如何将将二个有序数列合并。这个非常简单,只要从比较二个数列的第一个数,谁小
就先取谁,取了后就在对应数列中删除这个数。然后再进行比较,如果有数列为空,那直接将另一个数
列的数据依次取出即可。
(3)解决了上面的合并有序数列问题,再来看归并排序,其的基本思路就是将数组分成二组A,B,如
果这二组组内的数据都是有序的,那么就可以很方便的将这二组数据进行排序。如何让这二组组内数据
有序了?
可以将A,B组各自再分成二组。依次类推,当分出来的小组只有一个数据时,可以认为这个小组组内已
经达到了有序,然后再合并相邻的二个小组就可以了。这样通过先递归的分解数列,再合并数列就完成
了归并排序。
介绍一下,堆排序的原理是什么?
考察点:堆排序
参考回答:
堆排序分大顶堆和小顶堆,这里以大顶堆为例讲解。
堆排序就是把最大堆堆顶的最大数取出,将剩余的堆继续调整为最大堆,再次将堆顶的最大数取出,这
个过程持续到剩余数只有一个时结束。在堆中定义以下几种操作:
(1)最大堆调整(Max-Heapify):将堆的末端子节点作调整,使得子节点永远小于父节点。
(2)创建最大堆(Build-Max-Heap):将堆所有数据重新排序,使其成为最大堆。
(3)堆排序(Heap-Sort):移除位在第一个数据的根节点,并做最大堆调整的递归运算
image-20220716152732544
谈一谈,如何得到一个数据流中的中位数?
参考回答:
数据是从一个数据流中读出来的,数据的数目随着时间的变化而增加。如果用一个数据容器来保存从流
中读出来的数据,当有新的数据流中读出来时,这些数据就插入到数据容器中。
数组是最简单的容器。如果数组没有排序,可以用 Partition 函数找出数组中的中位数。在没有排序的
数组中插入一个数字和找出中位数的时间复杂度是 O(1)和 O(n)。
我们还可以往数组里插入新数据时让数组保持排序,这是由于可能要移动 O(n)个数,因此需要 O(n)时
间才能完成插入操作。在已经排好序的数组中找出中位数是一个简单的操作,只需要 O(1)时间即可完
成。
排序的链表时另外一个选择。我们需要 O(n)时间才能在链表中找到合适的位置插入新的数据。如果定义
两个指针指向链表的中间结点(如果链表的结点数目是奇数,那么这两个指针指向同一个结点),那么
可以在 O(1)时间得出中位数。此时时间效率与及基于排序的数组的时间效率一样。
如果能够保证数据容器左边的数据都小于右边的数据,这样即使左、右两边内部的数据没有排序,也可
以根据左边最大的数及右边最小的数得到中位数。如何快速从一个容器中找出最大数?用最大堆实现这
个数据容器,因为位于堆顶的就是最大的数据。同样,也可以快速从最小堆中找出最小数。
因此可以用如下思路来解决这个问题:用一个最大堆实现左边的数据容器,用最小堆实现右边的数据容
器。往堆中插入一个数据的时间效率是 O(logn)。由于只需 O(1)时间就可以得到位于堆顶的数据,因此
得到中位数的时间效率是 O(1)。
你知道哪些排序算法,这些算法的时间复杂度分别是多
少,解释一下快排?
image-20220716152941867
快排:快速排序有两个方向,左边的i下标一直往右走(当条件a[i] <= a[center_index]时),其中
center_index是中枢元素的数组下标,一般取为数组第0个元素。
而右边的j下标一直往左走(当a[j] > a[center_index]时)。如果i和j都走不动了,i <= j, 交换a[i]和a[j],重复上面的过程,直到i>j。交换a[j]和a[center_index],完成一趟快速排序。
6.7 堆与栈
请你解释一下,内存中的栈(stack)、堆(heap) 和静态区(static area) 的用法。
考察点:堆栈
参考回答:
堆区:专门用来保存对象的实例(new 创建的对象和数组),实际上也只是保存对象实例的属性值,属性的
类型和对象本身的类型标记等,并不保存对象的方法(方法是指令,保存在Stack中)
存储的全部是对象,每个对象都包含一个与之对应的class的信息。(class的目的是得到操作指令)
jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身.
一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。
栈区:对象实例在Heap 中分配好以后,需要在Stack中保存一个4字节的Heap内存地址,用来定位该对
象实例在Heap 中的位置,便于找到该对象实例。
每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都
存放在堆区
每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
由编译器自动分配释放 ,存放函数的参数值,局部变量的值等.
静态区/方法区:
方法区又叫静态区,跟堆一样,被所有的线程共享。方法区 包含所有的class和static变量。
方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化
的全局变量和未初始化的静态变量在相邻的另一块区域
说一说,heap和stack有什么区别。
参考回答:
1.heap是堆,stack是栈。
2.stack的空间由操作系统自动分配和释放,heap的空间是手动申请和释放的,heap常用new关键
字来分配。
3.stack空间有限,heap的空间是很大的自由区。在Java中,若只是声明一个对象,则先在栈内存
中为其分配地址空间,若再new一下,实例化它,则在堆内存中为其分配地址。
4.举例 数据类型 变量名;这样定义的东西在栈区。如:Object a =null; 只在栈内存中分配空间
new 数据类型();或者malloc(长度); 这样定义的东西就在堆区如:Object b =new Object(); 则在堆
内存中分配空间
介绍一下,堆与栈的不同是什么?
参考回答:
(1)Java的堆是一个运行时数据区,类的对象从中分配空间。通过比如:new等指令建立,不需要代
码显式的释放,由垃圾回收来负责。
优点:可以动态地分配内存大小,垃圾收集器会自动回收垃圾数据。
缺点:由于其优点,所以存取速度较慢。
(2)栈:
其数据项的插入和删除都只能在称为栈顶的一端完成,后进先出。栈中存放一些基本类型的 变量 和 对
象句柄。
优点:读取数度比堆要快,仅次于寄存器,栈数据可以共享。
缺点:比堆缺乏灵活性,存在栈中的数据大小与生存期必须是确定的。
举例:
String是一个特殊的包装类数据。可以用:
String str = new String("csdn");
String str = "csdn";
两种的形式来创建,第一种是用new()来新建对象的,它会在存放于堆中。每调用一次就会创建一个新
的对象。而第二种是先在栈中创建一个对String类的对象引用变量str,然后查找栈中有没有存
放"csdn",如果没有,则将"csdn"存放进栈,并令str指向”abc”,如果已经有”csdn” 则直接令str指向
“csdn”。
6.8队列
什么是Java优先级队列(Priority Queue)?
参考回答:
PriorityQueue是一个基于优先级堆的无界队列,它的元素是按照自然顺序(natural order)排序的。在创
建的时候,可以给它提供一个负责给元素排序的比较器。PriorityQueue不允许null值,因为他们没有自
然顺序,或者说他们没有任何的相关联的比较器。最后,PriorityQueue不是线程安全的,入队和出队
的时间复杂度是O(log(n))。
6.9 高级算法
请你讲讲LRU算法的实现原理?
参考回答:
注意:面试可能会让手写LRU算法!
1LRU(Least recently used,最近最少使用)算法根据数据的历史访问记录来进行淘汰数据,其核心
思想是“如果数据最近被访问过,那么将来被访问的几率也很高”,反过来说“如果数据最近这段时间一直
都没有访问,那么将来被访问的概率也会很低”,两种理解是一样的;常用于页面置换算法,为虚拟页式
存储管理服务。
2 达到这样一种情形的算法是最理想的: 每次调换出的页面是所有内存页面中最迟将被使用的;这可以
最大限度的推迟页面调换,这种算法,被称为理想页面置换算法。可惜的是,这种算法是无法实现的。
为了尽量减少与理想算法的差距,产生了各种精妙的算法,最近最少使用页面置换算法便是其中一个。
LRU 算法的提出,是基于这样一个事实:在前面几条指令中使用频繁的页面很可能在后面的几条指令中
频繁使用。反过来说,已经很久没有使用的页面很可能在未来较长的一段时间内不会被用到 。这个,就
是著名的局部性原理——比内存速度还要快的cache,也是基于同样的原理运行的。因此,我们只需要
在每次调换时,找到最近最少使用的那个页面调出内存。
LRU主要的两个函数 获取数据 get 和 写入数据 put 。
获取数据 get(key) - 如果关键字 (key) 存在于缓存中,则获取关键字的值(总是正数),否则返回
-1。
写入数据 put(key, value) - 如果关键字已经存在,则变更其数据值;如果关键字不存在,则插入该
组「关键字/值」。当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,
从而为新的数据值留出空间。
算法实现的关键
命中率:当存在热点数据时,LRU的效率很好,但偶发性的、周期性的批量操作会导致 LRU 命中
率急剧下降,缓存污染情况比较严重。
复杂度:实现起来较为简单。
存储成本:几乎没有空间上浪费。
代价:命中时需要遍历链表,找到命中的数据块索引,然后需要将数据移到头部。
为什么要设计后缀表达式,有什么好处?
考察点:逆波兰表达式
参考回答:
后缀表达式又叫逆波兰表达式,逆波兰记法不需要括号来标识操作符的优先级。
请你设计一个算法,用来压缩一段URL?考察点:MD5加密算法
参考回答:
该算法主要使用MD5 算法对原始链接进行加密(这里使用的MD5 加密后的字符串长度为32 位),然
后对加密后的字符串进行处理以得到短链接的地址。
谈一谈,id全局唯一且自增,如何实现?
考察点:SnowFlake雪花算法
参考回答;
SnowFlake雪花算法
雪花ID生成的是一个64位的二进制正整数,然后转换成10进制的数。64位二进制数由如下部分组成:
snowflake id生成规则
1位标识符:始终是0,由于long基本类型在Java中是带符号的,最高位是符号位,正数是0,负数
是1,所以id一般是正数,最高位是0。
41位时间戳:41位时间截不是存储当前时间的时间截,而是存储时间截的差值(当前时间截 - 开
始时间截 )得到的值,这里的的开始时间截,一般是我们的id生成器开始使用的时间,由我们程序
来指定的。
10位机器标识码:可以部署在1024个节点,如果机器分机房(IDC)部署,这10位可以由 5位机
房ID + 5位机器ID 组成。
12位序列:毫秒内的计数,12位的计数顺序号支持每个节点每毫秒(同一机器,同一时间截)产生
4096个ID序号
优点
简单高效,生成速度快。
时间戳在高位,自增序列在低位,整个ID是趋势递增的,按照时间有序递增。
灵活度高,可以根据业务需求,调整bit位的划分,满足不同的需求。
缺点
依赖机器的时钟,如果服务器时钟回拨,会导致重复ID生成。
在分布式环境上,每个服务器的时钟不可能完全同步,有时会出现不是全局递增的情况。
Design and implement a data structure for Least
Frequently Used (LFU) cache. It should support the
following operations: get and put. get(key) - Get the
value (will always be positive) of the key if the key
exists in the cache, otherwise return -1. put(key,
value) - Set or insert the value if the key is not
already present. When the cache reaches its
capacity, it should invalidate the least frequently
used item before inserting a new item. For the
purpose of this problem, when there is a tie (i.e.,
two or more keys that have the same frequency),
the least recently used key would be evicted. Could
you do both operations in O(1) time complexity?
考察点:LFU Cache
public class LFUCache {
private class Node{
int value;
ArrayList<Integer> set;
Node prev;
Node next;
public Node (int value ){
this.value = value;
this.set = new ArrayList<Integer> ();
this.prev = null;
this.next = null;
}
}
private class item{
int key;
int value;
Node parent ;
public item(int key ,int value, Node parent){
this.key = key ;
this.value = value;
this.parent = parent;
}
}
private HashMap<Integer, item> map;
private Node head,tail;
private int capacity;
// @param capacity, an integer
public LFUCache(int capacity) {
// Write your code here
this.capacity = capacity;
this.map = new HashMap <Integer,item> ();
this.head = new Node (0);
this.tail = new Node(Integer.MAX_VALUE);
head.next = tail;
tail.prev = head;
}
// @param key, an integer
// @param value, an integer
// @return nothing
public void set(int key, int value) {
// Write your code here
if (get(key) != -1 ) {
map.get(key).value = value;
return ;
}
if (map.size() == capacity ){
getLFUitem();
}
Node newpar = head.next;
if ( newpar.value != 1){
newpar = getNewNode(1,head,newpar);
}
item curitem = new item(key,value,newpar);
map.put(key,curitem);
newpar.set.add(key);
return;
}
public int get(int key) {
// Write your code here
if (!map.containsKey(key)){
return -1;
}
item cur = map.get(key);
Node curpar = cur.parent;
if (curpar.next.value == curpar.value + 1){
cur.parent= curpar.next;
cur.parent.set.add(key);
}else{
Node newpar =getNewNode(curpar.value + 1,curpar,curpar.next);
cur.parent = newpar;
newpar.set.add(key);
}
curpar.set.remove(new Integer(key));
if(curpar.set.isEmpty()){
deleteNode(curpar);
}
return cur.value;
}
private Node getNewNode (int value ,Node prev , Node next){
Node temp = new Node(value);
temp.prev = prev;
temp.next = next;
prev.next = temp;
next.prev = temp;
return temp;
}
private void deleteNode(Node temp){
temp.prev.next = temp.next;
temp.next.prev = temp.prev;
return ;}
private void getLFUitem(){
Node temp = head.next;
int LFUkey = temp.set.get(0);
temp.set.remove(0);
map.remove(LFUkey);
if (temp.set.isEmpty()){
deleteNode(temp);
}
return;
}
new 分配内存 如何实现的?****
malloc 底层 如何实现的?
参考回答:
new是操作符,而malloc是函数。
new在调用的时候 先分配内存,在调用构造函数,释放的时候 调用析构函数;而malloc没有构造
函数和析构函数。
malloc需要给定申请内存的大小,返回的指针需要强转; new会调用构造函数,不用指定内存的
大小,返回指针不用强转。
new可以被重载; malloc不行
new分配内存更直接和安全。
new发生 错误抛出异常,malloc 返回null
答案解析
malloc底层实现: 当开辟的空间小于 128K 时,调用 brk()函数;当开辟的空间大于 128K 时,调用
mmap()。malloc采用的是内存池的管理方式,以减少内存碎片。先申请大块内存作为堆区,然后将
堆区分为多个内存块。当用户申请内存时,直接从堆区分配一块合适的空闲快。采用隐式链表将所有空
闲块,每一个空闲块记录了一个未分配的、连续的内存地址。
new底层实现: 关键字new在调用 构造函数的时候实际上进行了如下的几个步骤:
创建一个新的对象
将构造函数的作用域赋值给 这个新的对象(因此this指向了这个新的对象)
执行构造函数中的代码(为这个新对象添加属性)
返回新对象
当我问你C++的内存布局时,你大概会回答:
“在C++中,内存区分为5个区,分别是堆、栈、自由存储区、全局/静态存储区、常量存储区”。
如果我接着问你自由存储区与堆有什么区别,你或许这样回答:
“malloc在堆上分配的内存块,使用free释放内存,而new所申请的内存则是在自由存储区上,使用delete来释放。”
这样听起来似乎也没错,但如果我接着问:
自由存储区与堆是两块不同的内存区域吗?它们有可能相同吗?
你可能就懵了。
事实上,网上看的很多博客,划分自由存储区与堆的分界线就是 n e w / d e l e t e new/deletenew/delete 与 m a l l o c / f r e e malloc/freemalloc/free 。然而,尽管C++标准没有要求,但很多编译器的 n e w / d e l e t e new/deletenew/delete 都是以 m a l l o c / f r e e malloc/freemalloc/free 为基础来实现的。
那么请问:借以 m a l l o c mallocmalloc 实现的 n e w newnew ,所申请的内存是在堆上还是在自由存储区上?
从技术上来说,堆(heap)是 C CC语言和操作系统的术语。堆是操作系统所维护的一块特殊内存,它提供了动态分配的功能,当运行程序调用 m a l l o c ( ) malloc()malloc() 时就会从中分配,稍后调用 f r e e freefree 可把内存交还。而自由存储是 C CC++中通过 n e w newnew 和 d e l e t e deletedelete 动态分配和释放对象的抽象概念,通过 n e w newnew 来申请的内存区域可称为自由存储区。
基本上,所有的C++编译器默认使用堆来实现自由存储,也即是缺省的全局运算符 n e w newnew 和 d e l e t e deletedelete 也许会按照 m a l l o c mallocmalloc 和 f r e e freefree 的方式来被实现,这时即由 n e w newnew 运算符分配的对象,说它在堆上也对,说它在自由存储区上也正确。但可以通过重载操作符,改用其他内存来实现自由存储,例如全局变量做的对象池,这时自由存储区就区别于堆了。 我们所需要记住的就是:
堆是操作系统维护的一块内存,而自由存储是C++中通过new与delete动态分配和释放对象的抽象概念。堆与自由存储区并不等价。
结论
自由存储 是C++中通过new与delete动态分配和释放对象的抽象概念,而堆(heap)是C语言和操作系统的术语,是操作系统维护的一块动态分配内存。
new所申请的内存区域在C++中称为自由存储区。藉由堆实现的自由存储,可以说new所申请的内存区域在堆上。
堆与自由存储区还是有区别的,它们并非等价。
————————————————
智能指针的实现原理、以及有哪些常用的指针?
虚函数的开销问题:?
一、表面上开销:
空间开销:
包含虚函数的类生成一个虚函数表,致使程序的二进制文件大小会相应的增大;
类的实例包含一个虚函数表指针,致使对象实例空间占用增加一个指针大小【32位系统,4个字节】
时间开销:
增加了一次内存寻址,通过虚函数表指针找到虚函数表,虽对程序性能有一些影响,但是影响并不大
二、背后的开销:【更深入的解释】
从汇编层生成的代码来看,普通函数与虚函数的区别是,普通函数是一个直接调用,而虚函数是一个间接调用,直
接调用与间接调用的区别就是 跳转地址是否确定, 直接调用的跳转地址是编译器确定的,而间接调用是运行到该指
令时从寄存器中取出地址然后跳转;
直接调用而言,是不存在分支跳转的,因为跳转地址是编译器确定的,CPU直接去跳转地址取后面的指令即可,不
存在分支预测,这样可以保证CPU流水线不被打断。
而对于间接寻址,由于跳转地址不确定,所以此处会有多个分支可能,这个时候需要分支预测器进行预测,如果分
支预测失败,则会导致流水线冲刷,重新进行取指、译码等操作,对程序性能有很大的影响。
深拷⻉与浅拷⻉
浅拷⻉只是拷⻉一个指针,并没有新开辟一个地址,拷⻉的指针 和原来的指针指向同一块地址,如果原来的指针所指向的 资源释放了,那么再释放浅拷⻉的指针的资源就会出现错误。
深拷⻉不仅拷⻉值,还开辟出一块新的空间用来存放新的值,即使原先的对象被析构掉,释放内存了也不会影响到深拷⻉得到的值。
在自己实现拷⻉赋值的时候,如果有指针变量的话是需要自己实现深拷⻉的。
inline与deifine的区别
主要区别:
宏在预编译时进行,只做 简单字符串替换。
内联函数 在编译时 直接将函数代码嵌入到目标代码中,省去 函数调用的开销来提高执行效率,并且进行 参数类型检查,**具有返回值,可以实现重载**。
内联函数适用场景:
使用宏定义的地方都可以使用inline函数。
作为 类成员接口函数来读写 类的私有成员或者保护成员,会提高效率。
为什么不能把所有的函数写成内联函数:
函数体内的函数体内有循环,函数执行时间要比函数调用开销大。。
函数体内有循环,函数执行时间要比函数调用开销大。
结构体内存对⻬的问题
1.1 概念
访问特定类型变量 的时候经常在特定的内存地址访问,这需要各种类型数据按照一定的规则在空间上排序,而不是顺序的一个接一个的排放
1.2 对⻬的作用和原因
各个硬件平台对存储空间的处理上有很大的不同。(一些平台对某些特定类型的数据只能从某些特定地址开始存取)
未对⻬:会导致访问变量时发生错误,以及读取效率上下降很多;
1.3 对⻬的原则
有效对⻬值N是最终用来决定数据存放地址方式的值,该数据的"存放起始地址%N=0。
数据结构中的数据变量都是按照定义的顺序来排放的,第一个数据变量的起始地址就是数据结构的起始地址。
结构体本身也要根据自身的有效对⻬值取整(结构体成员变量占用总⻓度需要是对结构体有效对⻬值的整数倍)
小总结:各变量要对⻬ + 结构体整体也要对⻬。
1.4 基础知识:
数据类型自身的对⻬值:对于char型数据,其自身对⻬值为1,对于short型为2,对于int,float,double类型,其自身对⻬值为4,单位字节。
结构体或者类的自身对⻬值:其成员中自身对⻬值最大的那个值。
指定对⻬:#pragma pack (value)时的指定对⻬值value。
数据成员、结构体和类的有效对⻬:自身对⻬值和 指定对⻬值中 小的那个值;
在结构体和类中成员函数中的非虚函数不占空间, 虚函数占一个指针的字节;
静态成员变量所有的 类对象 共享一份,在静态区域中,并不占用类对象的空间;
没有任何成员变量的类对象 占用一个字节的空间;
2. 指针与引用的区别
指针是一个变量,存储的是一个地址,引用跟原来的变量实质上是同一个东⻄,是原变量的别名;
指针可以有多级, 引用只有一级。
指针可以为空, 引用不能为NULL且在定义时必须初始化;
指针在初始化后可以改变指向, 而引用在初始化之后不可再改变;
sizeof 指针得到的是本指针的大小, sizeof引用 得到的是引用所指向变量的大小;
当把 指针作为参数进行传递时,也是将实参 的一个拷⻉传递给 形参,两者指向的地址相同,但不是同一个变量,在函数中改变这个变量的指向不影响实参,而引用却可以。
引用本质是一个指针,同样会占4字节内存;指针是具体变量,需要占用存储空间(具体情况还要具体分析)。
引用在声明时必须初始化为另一变量,一旦出现必须为typename refname &varname形式;
指针声明和定义可以分开,可以先只声明指针变量而不初始化,等用到时再指向具体变量。
引用一旦初始化之后就不可以再改变(变量可以被引用为多次,但引用只能作为一个变量引用);指针变量可以重新指向别的变量
10.不存在指向空值的引用,**必须有具体实体;但是存在**指向空值的指针。
2.操作系统
utf8编码占用多少字节?
Unicode的出现
Unicode 为世界上所有字符都分配了一个唯一的数字编号,这个编号范围从 0x000000 到 0x10FFFF (十六进制),有 110 多万,每个字符都有一个唯一的 Unicode 编号,这个编号一般写成 16 进制,在前面加上 U+。例如:“马”的 Unicode 是U+9A6C。
3、UTF-8
UTF-8 就是使用变长字节表示,顾名思义,就是使用的字节数可变,这个变化是根据 Unicode 编号的大小有关,编号小的使用的字节就少,编号大的使用的字节就多。使用的字节个数从 1 到 4 个不等。
UTF-8 的编码规则是:
① 对于单字节的符号,字节的第一位设为 0,后面的7位为这个符号的 Unicode 码,因此对于英文字母,UTF-8 编码和 ASCII 码是相同的。
② 对于n字节的符号(n>1),第一个字节的前 n 位都设为 1,第 n+1 位设为 0,后面字节的前两位一律设为 10,剩下的没有提及的二进制位,全部为这个符号的 Unicode 码 。
举个例子:比如说一个字符的 Unicode 编码是 130,显然按照 UTF-8 的规则一个字节是表示不了它(因为如果是一个字节的话前面的一位必须是 0),所以需要两个字节(n = 2)。
根据规则,第一个字节的前 2 位都设为 1,第 3(2+1) 位设为 0,则第一个字节为:110X XXXX,后面字节的前两位一律设为 10,后面只剩下一个字节,所以后面的字节为:10XX XXXX。
下面我们来具体看看具体的 Unicode 编号范围与对应的 UTF-8 二进制格式
img
那么对于一个具体的 Unicode 编号,具体怎么进行 UTF-8 的编码呢?
首先找到该 Unicode 编号所在的编号范围,进而可以找到与之对应的二进制格式,然后将该 Unicode 编号转化为二进制数(去掉高位的 0),最后将该二进制数从右向左依次填入二进制格式的 X 中,如果还有 X 未填,则设为 0 。
比如:“马”的 Unicode 编号是:0x9A6C,整数编号是 39532,对应第三个范围(2048 - 65535),其格式为:1110XXXX 10XXXXXX 10XXXXXX,39532 对应的二进制是 1001 1010 0110 1100,将二进制填入进入就为:
11101001 10101001 10101100 。
img
img
由于 UTF-8 的处理单元为一个字节(也就是一次处理一个字节),所以处理器在处理的时候就不需要考虑这一个字节的存储是在高位还是在低位,直接拿到这个字节进行处理就行了,因为大小端是针对大于一个字节的数的存储问题而言的。
综上所述,UTF-8、UTF-16、UTF-32 都是 Unicode 的一种实现。
2、UTF-16
UTF-16 使用变长字节表示
① 对于编号在 U+0000 到 U+FFFF 的字符(常用字符集),直接用两个字节表示。
② 编号在 U+10000 到 U+10FFFF 之间的字符,需要用四个字节表示。
同样,UTF-16 也有字节的顺序问题(大小端),所以就有 UTF-16BE 表示大端,UTF-16LE 表示小端。
linux 内部用什么编码?
进程的地址空间是多大?
socket--
编程流程是怎么样的?
如果没有 客户端连接的话,会阻塞在那个函数?
会阻塞在 accept 函数:
在这里插入图片描述
服务器 正在创建线程的时候 能够处理新来的连接吗?
客户端有多个请求 该怎么办?
虚拟内存:
虚拟内存
如果你是电子相关专业的,肯定在大学里捣鼓过单片机。
单片机是没有操作系统的,所以每次写完代码,都需要借助工具把程序烧录进去,这样程序才能跑起来。
另外,单片机的 CPU 是直接操作内存的「物理地址」。
在这种情况下,要想在内存中同时运行两个程序是不可能的。如果第一个程序在 2000 的位置写入一个新的值,将会擦掉第二个程序存放在相同位置上的所有内容,所以同时运行两个程序是根本行不通的,这两个程序会立刻崩溃。
操作系统是如何解决这个问题呢?
这里关键的问题是这两个程序都引用了绝对物理地址,而这正是我们最需要避免的。
我们可以把 进程所使用的地址「隔离」开来,即让操作系统为每个进程分配独立的一套「虚拟地址」,人人都有,大家自己玩自己的地址就行,互不干涉。但是有个前提每个进程都不能访问物理地址,至于虚拟地址最终怎么落到物理内存里,对进程来说是透明的,操作系统已经把这些都安排的明明白白了。
进程的中间层
操作系统会提供一种机制,将不同进程的虚拟地址和不同内存的物理地址映射起来。
如果程序要访问虚拟地址的时候,由操作系统转换成不同的物理地址,这样不同的进程运行的时候,写入的是不同的物理地址,这样就不会冲突了。
于是,这里就引出了两种地址的概念:
我们程序所使用的内存地址叫做虚拟内存地址(Virtual Memory Address)
实际存在硬件里面的空间地址叫物理内存地址(Physical Memory Address)。
操作系统引入了虚拟内存,进程持有的虚拟地址会通过 CPU 芯片中的内存管理单元(MMU)的映射关系,来转换变成物理地址,然后再通过物理地址访问内存,如下图所示:
img
操作系统是如何管理虚拟地址与物理地址之间的关系?
主要有两种方式,分别是内存分段和内存分页,分段是比较早提出的,我们先来看看内存分段。
内存分段
程序是由若干个逻辑分段组成的,如可由代码分段、数据分段、栈段、堆段组成。不同的段是有不同的属性的,所以就用分段(*Segmentation*)的形式把这些段分离出来。
分段机制下,虚拟地址和物理地址是如何映射的?
分段机制下的虚拟地址由两部分组成,段选择子和段内偏移量。
img
段选择子就保存在段寄存器里面。段选择子里面最重要的是段号,用作段表的索引。段表里面保存的是这个段的基地址、段的界限和特权等级等。
虚拟地址中的段内偏移量应该位于 0 和段界限之间,如果段内偏移量是合法的,就将段基地址加上段内偏移量得到物理内存地址。
在上面,知道了虚拟地址是通过段表与物理地址进行映射的,分段机制会把程序的虚拟地址分成 4 个段,每个段在段表中有一个项,在这一项找到段的基地址,再加上偏移量,于是就能找到物理内存中的地址,如下图
img
如果要访问段 3 中偏移量 500 的虚拟地址,我们可以计算出物理地址为,段 3 基地址 7000 + 偏移量 500 = 7500。
分段的办法很好,解决了程序本身不需要关心具体的物理内存地址的问题,但它也有一些不足之处:
第一个就是内存碎片的问题。
第二个就是内存交换的效率低的问题。
接下来,说说为什么会有这两个问题。
我们先来看看,分段为什么会产生内存碎片的问题?
我们来看看这样一个例子。假设有 1G 的物理内存,用户执行了多个程序,其中:
游戏占用了 512MB 内存
浏览器占用了 128MB 内存
音乐占用了 256 MB 内存。
这个时候,如果我们关闭了浏览器,则空闲内存还有 1024 - 512 - 256 = 256MB。
如果这个 256MB 不是连续的,被分成了两段 128 MB 内存,这就会导致没有空间再打开一个 200MB 的程序。
这里的内存碎片的问题共有两处地方:
这里的内存碎片的问题共有两处地方:
外部内存碎片,也就是产生了多个不连续的小物理内存,导致新的程序无法被装载;
内部内存碎片,程序所有的内存都被装载到了物理内存,但是这个程序有部分的内存可能并不是很常使用,这也会导致内存的浪费;
针对上面两种内存碎片的问题,解决的方式会有所不同。
解决外部内存碎片的问题就是内存交换。
可以把音乐程序占用的那 256MB 内存写到硬盘上,然后再从硬盘上读回来到内存里。不过再读回的时候,我们不能装载回原来的位置,而是紧紧跟着那已经被占用了的 512MB 内存后面。这样就能空缺出连续的 256MB 空间,于是新的 200MB 程序就可以装载进来。
这个内存交换空间,在 Linux 系统里,也就是我们常看到的 Swap 空间,这块空间是从硬盘划分出来的,用于内存与硬盘的空间交换。
再来看看,分段为什么会导致内存交换效率低的问题?
对于多进程的系统来说,用分段的方式,内存碎片是很容易产生的,产生了内存碎片,那不得不重新 Swap 内存区域,这个过程会产生性能瓶颈。
因为硬盘的访问速度要比内存慢太多了,每一次内存交换,我们都需要把一大段连续的内存数据写到硬盘上。
所以,如果内存交换的时候,交换的是一个占内存空间很大的程序,这样整个机器都会显得卡顿。
为了解决内存分段的内存碎片和内存交换效率低的问题,就出现了内存分页。
内存分页
分段的好处就是能产生连续的内存空间,但是会出现内存碎片和内存交换的空间太大的问题。
要解决这些问题,那么就要想出能少出现一些内存碎片的办法。另外,当需要进行内存交换的时候,让需要交换写入或者从磁盘装载的数据更少一点,这样就可以解决问题了。这个办法,也就是内存分页(Paging)。
分页是把整个虚拟和物理内存空间切成一段段固定尺寸的大小。这样一个连续并且尺寸固定的内存空间,我们叫页(Page)。在 Linux 下,每一页的大小为 4KB。
虚拟地址与物理地址之间通过页表来映射,如下图:
img
页表是*存储在内存里的,内存管理单元** (MMU*)就做将虚拟内存地址转换成物理地址的工作。
而当 进程访问的虚拟地址在页表中查不到时,系统会产生一个缺页异常,进入 系统内核空间 分配物理内存、更新进程页表,最后再返回用户空间,恢复进程的运行。
由于内存空间都是预先划分好的,也就不会像分段会产生间隙非常小的内存,这正是分段会产生内存碎片的原因。而采用了分页,那么释放的内存都是以页为单位释放的,也就不会产生无法给进程使用的小内存。
如果内存空间不够,操作系统会把其他正在运行的进程中的「最近没被使用」的内存页面给释放掉,也就是暂时写在硬盘上,称为换出(Swap Out)。一旦需要的时候,再加载进来,称为换入(Swap In)。所以,一次性写入磁盘的也只有少数的一个页或者几个页,不会花太多时间,内存交换的效率就相对比较高。
img
更进一步地,分页的方式使得我们在加载程序的时候,不再需要一次性都把程序加载到物理内存中。我们完全可以在进行虚拟内存和物理内存的页之间的映射之后,并不真的把页加载到物理内存里,而是只有在程序运行中,需要用到对应虚拟内存页里面的指令和数据时,再加载到物理内存里面去。
分页机制下,虚拟地址和物理地址是如何映射的?
在分页机制下,虚拟地址分为两部分,页号和页内偏移。页号作为页表的索引,页表包含物理页每页所在物理内存的基地址,这个基地址与页内偏移的组合就形成了物理内存地址,见下图。
总结一下,对于一个内存地址转换,其实就是这样三个步骤
把虚拟内存地址,切分成页号和偏移量;
根据页号,从页表里面,查询对应的物理页号;
直接拿物理页号,加上前面的偏移量,就得到了物理内存地址。
下面举个例子,虚拟内存中的页通过页表映射为了物理内存中的页,如下图:
img
这看起来似乎没什么毛病,但是放到实际中操作系统,这种简单的分页是肯定是会有问题的
简单的分页有什么缺陷吗?
有空间上的缺陷。
因为操作系统是可以同时运行非常多的进程的,那这不就意味着页表会非常的庞大。
在 32 位的环境下,虚拟地址空间共有 4GB,假设一个页的大小是 4KB(2^12),那么就需要大约 100 万 (2^20) 个页,每个「页表项」需要 4 个字节大小来存储,那么整个 4GB 空间的映射就需要有 4MB 的内存来存储页表。
这 4MB 大小的页表,看起来也不是很大。但是要知道每个进程都是有自己的虚拟地址空间的,也就说都有自己的页表。
那么,100 个进程的话,就需要 400MB 的内存来存储页表,这是非常大的内存了,更别说 64 位的环境了。
多级页表
要解决上面的问题,就需要采用一种叫作多级页表(Multi-Level Page Table)的解决方案。
在前面我们知道了,对于单页表的实现方式,在 32 位和页大小 4KB 的环境下,一个进程的页表需要装下 100 多万个「页表项」,并且每个页表项是占用 4 字节大小的,于是相当于每个页表需占用 4MB 大小的空间。
我们把这个 100 多万个「页表项」的单级页表再分页,将页表(一级页表)分为 1024 个页表(二级页表),每个表(二级页表)中包含 1024 个「页表项」,形成二级分页。如下图所示:
img
你可能会问,分了二级表,映射 4GB 地址空间就需要 4KB(一级页表)+ 4MB(二级页表)的内存,这样占用空间不是更大了吗?
当然如果 4GB 的虚拟地址全部都映射到了物理内存上的话,二级分页占用空间确实是更大了,但是,我们往往不会为一个进程分配那么多内存。
其实我们应该换个角度来看问题,还记得计算机组成原理里面无处不在的局部性原理么?
每个进程都有 4GB 的虚拟地址空间,而显然对于大多数程序来说,其使用到的空间远未达到 4GB,因为会存在部分对应的页表项都是空的,根本没有分配,对于已分配的页表项,如果存在最近一定时间未访问的页表,在物理内存紧张的情况下,操作系统会将页面换出到硬盘,也就是说不会占用物理内存。
如果使用了二级分页,一级页表就可以覆盖整个 4GB 虚拟地址空间,但如果某个一级页表的页表项没有被用到,也就不需要创建这个页表项对应的二级页表了,即可以在需要时才创建二级页表。做个简单的计算,假设只有 20% 的一级页表项被用到了,那么页表占用的内存空间就只有 4KB(一级页表) + 20% * 4MB(二级页表)= 0.804MB,这对比单级页表的 4MB 是不是一个巨大的节约?
————————————————
那么为什么不分级的页表就做不到这样节约内存呢?我们从页表的性质来看,保存在内存中的页表承担的职责是将虚拟地址翻译成物理地址。假如虚拟地址在页表中找不到对应的页表项,计算机系统就不能工作了。所以页表一定要覆盖全部虚拟地址空间,不分级的页表就需要有 100 多万个页表项来映射,而二级分页则只需要 1024 个页表项(此时一级页表覆盖到了全部虚拟地址空间,二级页表在需要时创建)。
我们把二级分页再推广到多级页表,就会发现页表占用的内存空间更少了,这一切都要归功于对局部性原理的充分应用。
对于 64 位的系统,两级分页肯定不够了,就变成了四级目录,分别是:
全局页目录项 PGD(Page Global Directory);
上层页目录项 PUD(Page Upper Directory);
中间页目录项 PMD(Page Middle Directory);
页表项 PTE(Page Table Entry);
img
TLB
多级页表虽然解决了空间上的问题,但是虚拟地址到物理地址的转换就多了几道转换的工序,这显然就降低了这俩地址转换的速度,也就是带来了时间上的开销。
程序是有局部性的,即在一段时间内,整个程序的执行仅限于程序中的某一部分。相应地,执行所访问的存储空间也局限于某个内存区域。
img
我们就可以利用这一特性,把最常访问的几个页表项存储到访问速度更快的硬件,于是计算机科学家们,就在 CPU 芯片中,加入了一个专门存放程序最常访问的页表项的 Cache,这个 Cache 就是 TLB(Translation Lookaside Buffer) ,通常称为页表缓存、转址旁路缓存、快表等。
img
在 CPU 芯片里面,封装了内存管理单元(Memory Management Unit)芯片,它用来完成地址转换和 TLB 的访问与交互。
有了 TLB 后,那么 CPU 在寻址时,会先查 TLB,如果没找到,才会继续查常规的页表。
TLB 的命中率其实是很高的,因为程序最常访问的页就那么几个。
段页式内存管理
内存分段和内存分页并不是对立的,它们是可以组合起来在同一个系统中使用的,那么组合起来后,通常称为段页式内存管理。
img
段页式内存管理实现的方式:
先将程序划分为多个有逻辑意义的段,也就是前面提到的分段机制;
接着再把每个段划分为多个页,也就是对分段划分出来的连续空间,再划分固定大小的页
这样,地址结构就由段号、段内页号和页内位移三部分组成。
用于段页式地址变换的数据结构是每一个程序一张段表,每个段又建立一张页表,段表中的地址是页表的起始地址,而页表中的地址则为某页的物理页号,如图所示:
img
段页式地址变换中要得到物理地址须经过三次内存访问:
第一次访问段表,得到页表起始地址;
第二次访问页表,得到物理页号;
第三次将物理页号与页内位移组合,得到物理地址。
可用软、硬件相结合的方法实现段页式地址变换,这样虽然增加了硬件成本和系统开销,但提高了内存的利用率。
Linux 内存管理
那么,Linux 操作系统采用了哪种方式来管理内存呢?
在回答这个问题前,我们得先看看 Intel 处理器的发展历史。
早期 Intel 的处理器从 80286 开始使用的是段式内存管理。但是很快发现,光有段式内存管理而没有页式内存管理是不够的,这会使它的 X86 系列会失去市场的竞争力。因此,在不久以后的 80386 中就实现了页式内存管理。也就是说,80386 除了完成并完善从 80286 开始的段式内存管理的同时还实现了页式内存管理。
但是这个 80386 的页式内存管理设计时,没有绕开段式内存管理,而是建立在段式内存管理的基础上,这就意味着,页式内存管理的作用是在由段式内存管理所映射而成的地址上再加上一层地址映射。
由于此时由段式内存管理映射而成的地址不再是“物理地址”了,Intel 就称之为“线性地址”(也称虚拟地址)。于是,段式内存管理先将逻辑地址映射成线性地址,然后再由页式内存管理将线性地址映射成物理地址。
————————————————
img
这里说明下逻辑地址和线性地址:
程序所使用的地址,通常是没被段式内存管理映射的地址,称为逻辑地址;
通过段式内存管理映射的地址,称为线性地址,也叫虚拟地址;
逻辑地址是「段式内存管理」转换前的地址,线性地址则是「页式内存管理」转换前的地址。
了解完 Intel 处理器的发展历史后,我们再来说说 Linux 采用了什么方式管理内存?
Linux 内存主要采用的是页式内存管理,但同时也不可避免地涉及了段机制。
这主要是上面 Intel 处理器发展历史导致的,因为 Intel X86 CPU 一律对程序中使用的地址先进行段式映射,然后才能进行页式映射。既然 CPU 的硬件结构是这样,Linux 内核也只好服从 Intel 的选择。
但是事实上,Linux 内核所采取的办法是使段式映射的过程实际上不起什么作用。也就是说,“上有政策,下有对策”,若惹不起就躲着走。
Linux 系统中的每个段都是从 0 地址开始的整个 4GB 虚拟空间(32 位环境下),也就是所有的段的起始地址都是一样的。这意味着,Linux 系统中的代码,包括操作系统本身的代码和应用程序代码,所面对的地址空间都是线性地址空间(虚拟地址),这种做法相当于屏蔽了处理器中的逻辑地址概念,段只被用于访问控制和内存保护。
我们再来瞧一瞧,Linux 的虚拟地址空间是如何分布的?
在 Linux 操作系统中,虚拟地址空间的内部又被分为内核空间和用户空间两部分,不同位数的系统,地址空间的范围也不同。比如最常见的 32 位和 64 位系统,如下所示:
img
通过这里可以看出:
32 位系统的内核空间占用 1G,位于最高处,剩下的 3G 是用户空间;
64 位系统的内核空间和用户空间都是 128T,分别占据整个内存空间的最高和最低处,剩下的中间部分是未定义的。
再来说说,内核空间与用户空间的区别:
进程在用户态时,只能访问用户空间内存;
只有进入内核态后,才可以访问内核空间的内存;
虽然每个进程都各自有独立的虚拟内存,但是每个虚拟内存中的内核地址,其实关联的都是相同的物理内存。这样,进程切换到内核态后,就可以很方便地访问内核空间内存。
img
接下来,进一步了解虚拟空间的划分情况,用户空间和内核空间划分的方式是不同的,内核空间的分布情况就不多说了。
我们看看用户空间分布的情况,以 32 位系统为例,我画了一张图来表示它们的关系:
虚拟内存空间划分
通过这张图你可以看到,用户空间内存,从低到高分别是 6 种不同的内存段:
程序文件段(.text),包括二进制可执行代码;
已初始化数据段(.data),包括静态常量;
未初始化数据段(.bss),包括未初始化的静态变量;
堆段,包括动态分配的内存,从低地址开始向上增长;
文件映射段,包括动态库、共享内存等,从低地址开始向上增长(跟硬件和内核版本有关);
栈段,包括局部变量和函数调用的上下文等。栈的大小是固定的,一般是 8 MB。当然系统也提供了参数,以便我们自定义大小;
在这 7 个内存段中,堆和文件映射段的内存是动态分配的。比如说,使用 C 标准库的 malloc() 或者 mmap() ,就可以分别在堆和文件映射段动态分配内存。
1、系统调用:
(1)操作系统提供了一组特殊接口——系统调用
通过这组接口 用户程序 可以使用操作系统内核提供的各种功能。例如分配内存、创建进程、实现进程之间的通信
等。
(2)用户程序 向操作系统提出请求的接口 就是 系统调用。
所有的操作系统都会提供系统调用接口,只不过不同的操作系统提供的系统调用接口各不相同。
(3)系统调用按照功能分类
进程控制、进程间通信、文件系统控制、存储管理、网络管理、套接字控制、用户管理等。
2、库函数:
对系统调用的一种封装,因为系统调用是面对的是操作系统,系统包括Linux、Windows等,如果直接系统调用,会影响程序的移植性,所以这里使用了库函数。
用户编程接口API:
前面提到利用系统调用接口程序可以访问各种资源,但在实际开发中程序并不直接使用系统调用接口,而是使用用户编程接口(API)【各种库(最重要的就是C库)中的函数】。
不直接使用系统调用接口原因如下:
系统调用接口功能非常简单,无法满足程序的需求。
不同操作系统的系统调用接口不兼容,程序移植时工作量大。
⻚面淘汰算法
1、淘汰策略(⻚式系统)
(1)置换算法
当主存块已全部用完 ,添加新的一⻚进主存,选择淘汰哪一⻚的规则
(2)颠簸(抖动)
导致系统效率急剧下降的主存和辅存之间的频繁⻚面置换现象。
(3)缺⻚(中断)率
f' = f/a(缺⻚次数/访问总次数)
(4)固定空间⻚面调度
系统为每一个进入主存的程序分配的主存块数m是固定的;
2、最佳算法(OPT算法)
每次选择的淘汰⻚面将是以后永不使用,或者在最⻓时间内都不再访问的⻚面(保证最低的缺⻚率)
该算法是无法实现的
原因:操作系统 无法提前判断⻚面访问序列一般该算法作为衡量各种具体算法优劣的标准
淘汰时参考的是未来将要执行的,但是只有在进程执行过程中才能知道接下来会访问到的是哪个⻚面
3、先进先出算法(FIFO算法)
每次淘汰选择在主存中居留时间最⻓(即进入最早)的一⻚淘汰。
实现的方法:
把调入内存的⻚面 根据调入的先后顺序排成一个队列,需要换出⻚面时选择队头;⻚面队列的最大⻓度取决于系统
为进程分配了多少个内存块。
Belady异常:
当为进程分配的物理块数增大时,缺⻚次数不降反增的异常现象。(只有FIFO算法有该现象)
缺点:
该算法与进程实际运行时的规律不适用,因为先进入的⻚面可能经常被访问,算法性能较差。
4、最近最久未使用淘汰算法(Least Recently Used)(LRU算法)
每次淘汰的⻚面时 最近最久未使用的⻚面
实现方法:
(1)计数器:
赋予 每个⻚面对应的⻚表项中, 用访问字段 记录 该⻚面 自上次被访问以所经历的时间t(需要硬件支持,实现困难,开销大)
(2)堆栈(软件实现)
利用 那个栈 来登记主存中可淘汰的⻚号。每当一个⻚面被访问过,就立即将它的⻚号记载⻚号栈的顶部,而将栈中原有的⻚号依次下移。(淘汰的⻚面对应栈底⻚面)
5、LRU近似算法(软硬相结合)
在一个存储块有一个引用位:
当某块中的⻚面
版权声明:本文标题:互联网大厂知识点整理 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/biancheng/1725920354a1030540.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论