C++ 万字长文第二篇---拿下字节面试
2020-11-05 23:09
shared_ptr 指针的实现
template<typename T> class Shared_ptr {
private:
T *ptr;
int *use_count; // 用指针保证指向同一块地址
public:
Shared_ptr() {
ptr = nullptr;
use_count = nullptr;
cout << "Created" << endl;
}
Shared_ptr(T *p){
ptr = p;
use_count = new int(1);
cout << "Created" << endl;
}
Shared_ptr(const Shared_ptr
&other) { ptr = other.ptr;
++(*other.use_count);
use_count = other.use_count;
cout << "Created" << endl;
}
Shared_ptr
& operator=(const Shared_ptr &other) { if(this == &other) return *this;
(*other.use_count)++;
if(ptr && --(*use_count) == 0) {
delete ptr;
delete use_count;
}
ptr = other.ptr;
use_count = other.use_count;
return *this;
}
T& operator*() { // 返回指针的引用
return *ptr;
}
T* operator->() {
return ptr;
}
int get_use_count() {
if(use_count == nullptr) return 0;
return *use_count;
}
~Shared_ptr() {
if(ptr && --(*use_count) == 0) {
delete ptr;
delete use_count;
cout << "Destroy" << endl;
}
}
};
构造函数不可以是虚函数
从存储空间角度,虚函数对应一个虚函数表,而指向虚函数表的虚函数指针是存储区对象内存内的。如果构造函数是虚函数,则需要通过虚函数表来调用,而对象还没有构造出来,无法找到虚函数表。 从使用角度,虚函数主要用于信息不全的情况下,使子类重写的函数能得到对应的调用。构造函数本身就是要初始化对象,所以用虚函数没有意义。
平衡树(AVL)、伸展树(Splay)、红黑树
平衡树: 优点:查找、插入、删除最坏复杂度都是 O(logN)。操作简单。 缺点:插入、删除的旋转成本不菲。删除操作后,最多旋转 O(logN) 次,若频繁进行插入、删除操作,得不偿失 伸展树 优点:局部性强: 1)刚刚访问过的节点,可能在不久后再次被访问到。2) 将要访问的节点,可能在不久之前刚刚被访问的节点附近。 缺点:不能保证单次最坏情况的出现,不适用于敏感场合。复杂度不好分析,均摊 O(logN)。 红黑树 优点:查找、插入、删除的复杂度都是 O(logN)。插入之后最多旋转 2 次,删除之后最多旋转 1 次能够再次达到平衡状态。 缺点:左右子树高度相差比 AVL 大。
AVL 和 红黑树比较
AVL 是严格的平衡树,因此在插入/删除节点时,根据不同的情况,旋转的次数比红黑树多。
红黑树是弱平衡的,用非严格的平衡换取插入/删除节点时旋转次数的降低。
两者都是平衡树,那么查找的时候,查找效率就会和树的高度有关,所以 AVL 会比红黑树更优。
有了 AVL 为什么还要红黑树
虽然 AVL 解决的二叉查找树退化成链表的情况,但是平衡树要求每个节点左子树和右子树高度相差最多为 1。这个要求太严格了,导致插入/删除的时候需要不断的旋转来达到这个要求。而红黑树不会频繁的破坏规则,不用频繁的调整,红黑树是一种不大严格的平衡树,但是换来了效率的提高,是一种折中的方案。
红黑树
面试常问:什么是红黑树?
wiki红黑树
红黑树性质
节点一定是红色或黑色。 根节点是黑色。 每个叶子节点都是黑色的空节点(NIL)。 每个红色节点的两个子节点都是黑色。 从任一节点到其每个叶子节点的所有路径上都包含相同的黑色节点数。
这些性质强制了红黑树从根到叶子的最长路径不会超过最短路径的两倍。性质 4 导致了路径不能有两个相连的红色节点,最短的可能路径都是黑色节点,最长的可能路径有交替的红色和黑色节点。因为根据性质 5 所有最长的路径都有相同数目的黑色节点,这就表明了没有路径能多于任何其他路径的两倍长。
红黑树的扩展
可以给每个阶段加上 size 域,表示以节点 x 为根的子树的节点数。
size[x] = size[x->left]+size[x->right]
通过 size 就可以获得比 x 小的节点数目和第 x 小的节点。
i++ 和 ++i 的区别
++i 返回对象的引用,i++ 必须产生一个临时对象保存更改前对象的值并返回,所以导致在大对象的时候产生的比较大的复制开销,效率较低。
// ++i
int& int::operator++() {
*this += 1;
return *this;
}
// i++
const int int::operator++(int) {
int old_value = *this;
++(*this);
return old_value;
}
拷贝构造函数
Node a;
Node b(a);
Node c = a;
这里的 b 和 c 都是一开始是不存在的,是通过 a 对象来构造和初始化的。
拷贝构造函数重载形式:
Node (const Node &other) {
}
如果用户没有自定义拷贝构造函数并且用到了拷贝构造函数,则编译器会生成默认的拷贝构造函数。
在 C++ 中,这三种情况下拷贝构造函数会被使用:
一个对象以值传递的形式传入函数内。 一个对象以值传递的形式从函数返回。 一个对象通过另一个对象初始化。
优点:可以很容易的复制对象。
缺点:对象的隐式拷贝是 C++ 中是错误和性能问题的来源之一。它也降低了代码的可读性,并使得对象子程序中的传递和改变变得难以跟踪。
赋值函数
Node a;
Node b;
b = a;
这里的 b 已经存在的,在通过 a 赋值给 b。
赋值函数重载形式:
Node& operator=(const Node &other) {
}
拷贝构造函数和赋值函数区别
拷贝构造函数是在对象初始化时,分配一块空间并初始化,而赋值函数是对已经分配空间的对象进行赋值操作。 实现上,拷贝构造函数是构造函数,通过参数的对象初始化产生一个对象。赋值函数则是把一个对象赋值给另一个对象,需要先判断两个对象是否是同一个对象,若是则什么都不做,直接返回,若不是则需要先释放原对象内存,在赋值。(可以参考 shared_ptr 实现)
总结:
对象不存在,没有通过别的对象来初始化,就是构造函数。 对象不存在,通过别的对象来初始化,就是拷贝构造函数。 对象存在,通过别的对象来初始化,就是赋值函数。
虚函数和内联函数
内联函数通常就是将它在调用处 "内敛的" 展开,消除反复调用的额外开销,但是如果函数太长,会使代码臃肿,反而降低效率。
虚函数可以是内联函数,无论是显式还是隐式,inline 都只是一个申请,最终由编译器决定是否内联。所以可以用 inline 修饰虚函数,但虚函数表现多态性时,不可以内联,只有当知道调用哪个类的函数时,才可以是内联。
空类的大小
在 C++ 中规定类的大小不为 0,空类大小为 1,当类不包含虚函数和非静态成员时,其对象大小也为 1。若存在虚函数,则需要存储一个虚函数指针大小,在 32 位上为 4 字节。
结构体字节对齐
class A {
};
class B{
public:
A x;
};
sizeof(B) = 1
class B{
public:
inline virtual fun() {
}
};
sizeof(B) = 4
class B{
public:
A x;
inline virtual fun() {
}
};
sizeof(B) = 8
可以发现最后一个的 sizeof 并不是单纯的 1+4=5,而直接变成了 8,因为存在结构体的字节对齐规则。
结构体字节对齐的根本原因:1) 移植性更好,某些平台只能在特定的地址访问特定的数据。2) 提高存取数据的速度,CPU 通常按块读取数据会更加快速。
结构体字节对齐原则:
无 #pragma pack 结构内部各成员首地址必然是自身大小的整数倍。 sizeof 最终结果必然是结构内部最大成员的整数倍,不够补齐 有 #pragma pack(n) 结构内部各成员首地址必然是 min(n, 自身大小) 的整数倍。 sizeof 最终结果必然是 min(n, 结构内部最大成员) 的整数倍,不够补齐。
using namespace std;
class A {
char a;
int b;
short c;
};
class B {
int a;
char b;
short c;
};
int main() {
cout << sizeof(A) << endl; // 12
cout << sizeof(B) << endl; // 8
return 0;
}
造成不同结果的原理在于:
对于 class A 来说,内部字节为
a | b | c |
---|---|---|
1*** | 1111 | 11** |
对于 class B 来说,内部字节为
a | b | c |
---|---|---|
1111 | 1* | 11 |
有 static、virtual 之类的一个类的内存分布
static 修饰成员变量 静态成员变量在 全局存储区 分配内存,本类的所有对象共享,在还没生成类对象之前也可以使用。 static 修饰成员函数 静态成员函数在 代码区 分配内存。静态成员函数和非静态成员函数的区别在于非静态成员函数存在 this 指针,而静态成员函数不存在,所以静态成员函数没有类对象也可以调用。 virtual 虚函数表存储在常量区,也就是只读数据段 虚函数指针存储在对象内,如果对象是局部变量,则存储在栈区内。
使用宏定义求结构体成员偏移量
/*
(TYPE*)0 将零转型成 TYPE 类型指针
((TYPE*)0->MEMBER) 访问结构体中的成员
&((TYPE*)0->MEMBER) 取出数据成员地址,也就是相对于零的偏移量
(size_t) & ((TYPE*)0)->MEMBER) 将结果转换成 size_t 类型。
*/
struct Node {
char a;
short b;
double c;
int d;
};
int main() {
printf("%d\n", offsetof(Node, a));
printf("%d\n", offsetof(Node, b));
printf("%d\n", offsetof(Node, c));
printf("%d\n", offsetof(Node, d));
return 0;
}
size_t 在可以理解成 unsigned\ \ int,在不同平台下被 typedef 成不同类型。
C++ 中哪些函数不可以是虚函数
普通函数(非成员函数):普通函数不属于类成员,不能被继承。普通函数只能被重载,不能被重写,因此声明成虚函数没有意义。 构造函数:构造函数本来就是为了初始化对象而存在的,没有定义为虚函数的必要。而且对象还没构造出来,不存在虚函数指针,也无法成为虚函数。 内联成员函数:内联函数是为了在代码中直接展开,减少调用函数的代价,是在编译的时候展开。而虚函数需要动态绑定,这不可能统一。只有当编译器知道调用的是哪个类的函数时,才可以展开。 静态成员函数:对于所有对象都共享一个函数,没有动态绑定的必要。 友元函数:C++ 不支持友元函数的继承,自然不能是虚函数。
手撕堆排序
[堆排序的思路以及代码的实现
](https://blog.csdn.net/qq_36573828/article/details/80261541)
令 k = log_2n
初始化堆复杂度 O(N) 。分析:假设每一次都到叶子的父节点
排序重建堆复杂度 。分析:假设每一次都到叶子节点。
void adjust(vector<int>& nums, int i, int n) {
while(2*i+1 < n) {
int j = 2*i+1;
if(j+1
1 ]) j++;if(nums[i] > nums[j]) break;
swap(nums[i], nums[j]);
i = j;
}
}
vector<int> sortArray(vector<int>& nums) {
if(nums.size() <= 1) return nums;
int n = nums.size();
for(int i=n/2-1; i>=0; i--) adjust(nums, i, n); // 初始化堆
for(int i=n-1; i>0; i--) { // 排序重建堆
swap(nums[i], nums[0]);
adjust(nums, 0, i);
}
return nums;
}
main 函数前后还会执行什么
全局对象的构造在 main 函数之前完成,全局对象的析构在 main 函数之后完成。
const 和 define
define 在预编译阶段起作用,const 在编译、运行的时候起作用。 define 只是简单的替换功能,没有类型检查功能,const 有类型检查功能,可以避免一些错误。 define 在预编译阶段就替换掉了,无法调试,const 可以通过集成化工具调试。
inline 和 define
inline 在编译时展开,define 在预编译时展开。 inline 可以进行类型安全检查,define 只是简单的替换。 inline 是函数,define 不是函数。 define 最好用括号括起来,不然会产生二义性,inline 不会。 inline 是一个建议,可以不展开,define 一定要展开。
inline 函数的要求
含有递归调用的函数不能设置为 inline 循环语句和 switch 语句,无法设置为 inline inline 函数内的代码应很短小。最好不超过5行
声明和定义
变量定义:为变量分配空间,还可以为变量指定初始值。 变量声明:向程序表明变量的类型和名字,但不分配空间。可以通过 extern 关键字来声明而不定义,extern 告诉编译器变量在别的地方定义了。
定义也是声明,声明不是定义。例如: extern int i 声明且不定义 i 变量,int i 声明且定义了 i 变量。 声明有初始值时,为当成定义 extern int i=10 此时看成定义了 i 变量。 函数的声明和定义,区别在于是否带有花括号。
面向过程和面向对象
面向过程就是分析出解决问题所需要的步骤,然后用函数把步骤一步一步的实现。优点在于性能高。
面向对象就是构成问题事务分解成各个对象,建立对象的目的不是为了完成一个步骤,而是为了描述某个对象在整个解决问题的步骤中的行为。优点在于易维护、易复用、易扩展,使系统更加灵活。
堆区和自由存储区
基本上所有的 C++ 编译器默认使用堆来实现自由存储,也就是缺省的 new/delete 是通过 malloc/free 方式来实现的,这时候可以说他从堆上分配内存,也可以说他从自由存储区分配内存。但程序员可以通过重载操作符,改用其他内存实现自由存储。
堆区是操作系统维护的一块内存,而自由存储区是 C++ 中用于 new/delete 动态分配和释放对象的 抽象概念。
堆区和栈区
堆区 | 栈区 | |
---|---|---|
管理方式 | 由程序员控制 | 系统主动分配 |
空间大小 | 空间很大,可以达到4G | 默认1M |
碎片问题 | 频繁的 malloc/free 会造成大量碎片 | 不会存在这个问题 |
生长方向 | 向着内存地址增加的方向 | 向着内存地址减小的方向 |
分配效率 | 速度比较慢 | 速度比较快 |
#ifndef 和 #endif 的作用
在大的软件工程中,可能多个文件包含同一个头文件,当这些文件链接成可执行文件的时候,就会造成大量的 "重定义" 错误,可以通过 #ifndef 来避免这个错误。
#ifndef _A_H
#define _A_H
#endif
这样就可以保证 a.h 被定义一次。
explicit 关键字
类的构造函数存在隐式转换,如果想要避免这个功能,就可以通过 explicit 关键字来将构造函数声明成显式的。
菱形继承
class A {
public:
void fun() {
cout << "!!!" << endl;
}
};
class B : public A{
};
class C : public A{
};
class D : public B, public C {
};
int main() {
// freopen("in", "r", stdin);
D d;
d.fun();;
return 0;
}
像上面的继承关系,当继承关系形成菱形时,D 中保存了两份 A,在调用 d.fun() 时就会出现调用不明确的问题。
这种情况由两种解决方法:
使用域限定访问的函数。也就是将 d.fun 修改成 d.B::fun 或者 d.C::fun。 第二种方法就是使用虚继承。虚继承解决了从不同途径继承来的同名数据成员在内存中有不同的拷贝造成数据不一致的问题,将共同基类设置成虚基类,这时从不同路径继承过来的同名数据成员在内存中就只有一个拷贝。操作方法就是在 B 和 C 的继承处加上 virtual 修饰。
虚继承底层实现一般通过虚基类指针和虚基类表实现。每个虚继承的子类都有一个虚基类指针和一个虚基类表,虚表中记录了虚基类与本类的偏移地址。通过偏移地址,这样就找到了虚基类成员,节省了存储空间。
weak_ptr 指向的对象被销毁
首先 weak_ptr 是一种不控制对象生存周期的智能指针,它指向一个 shared_ptr 管理的对象。一旦最后一个指向对象的 shared_ptr 被销毁,那么无论是否有 weak_ptr 指向该对象,都会释放资源。
weak_ptr 不能直接指向对象,需要先调用 lock,而 lock 会先判断该对象是否还存在。
如果存在 lock 就会返回一个指向该对象的 shared_ptr,并且该对象的 shared_ptr 的引用计数加一。
如果在 weak_ptr 获得过程中,原本的所有 shared_ptr 被销毁,那么该对象的生命周期会延长到这个临时 shared_ptr 销毁为止。
lambda 表达式
lambda 表达式定义一个匿名函数,并且可以捕获一定范围内的变量,基本格式如下:
[capture](params) mutable ->ReturnType {statement}
[capture]:捕获列表,可以捕获上下文的变量来供 lambda 函数使用 [var]:值传递的方式捕获 var [&var]:引用传递的方式捕获 var [=]:值传递的方式捕获父作用域的所有变量。 [&]:引用传递的方式捕获父作用域的所有变量。 [this]:值传递的方式捕获当前 this 指针。 (params):参数列表,和普通函数参数列表一致,如果不传参数可以和 () 一起忽略。 mutable 修饰符号,默认情况下,lambda 表达式是一个 const 函数,可以用 mutable 取消他的常量性。若使用 mutable 修饰,参数列表不可省略。 **->**ReturnType:返回值类型。若是 void,可以省略。 {statement}:函数主体,和普通函数一样。
lambda 表达式优点在于代码简洁,容易进行并行计算。缺点在于在非并行计算中,效率未必有 for 循环快,并且不容易调试,对于没学过的程序员来说可读性差。
using namespace std;
int main() {
vector<int> g{9, 2, 1, 2, 5, 6, 2};
int ans = 1;
sort(g.begin(), g.end(), [](int a, int b) ->bool{return a>b;});
for_each(g.begin(), g.end(), [&ans](int x){cout << x << " ", ans = ans*x;});
cout << "\nmul = " << ans << endl;
return 0;
}
/*
9 6 5 2 2 2 1
mul = 2160
*/
存在全局变量和局部变量时,访问全局变量
可以通过全局作用符号 :: 来完成
int a = 10;
int main() {
// freopen("in", "r", stdin);
int a = 20;
cout << a << endl;
cout << ::a << endl;
return 0;
}
全局变量的初始化的顺序
同一文件中的全局变量按照声明顺序,不同文件之间的全局变量初始化顺序不确定。
如果要保证初始化次序的话,需要通过在函数使用静态局部变量并返回来实现。
class FileSystem{...};
FileSystem & tfs(){
static FileSystem fs;//定义并初始化一个static对象
return fs;
}
浅拷贝和深拷贝
浅拷贝:源对象和拷贝对象共用一份实体,仅仅是引用的变量名不同。对其中任意一个修改,都会影响另一个对象。 深拷贝:源对象和拷贝对象相互独立。对其中一个对象修改,不会影响另一个对象。 两个对象指向同块内存,当析构函数释放内存时,会引起错误。
从这个例子可以看出,b 通过默认拷贝函数进行初始化,然而进行的是浅拷贝,导致对 a 进行修改的时候,b 的存储值也被修改。
struct Node {
char* s;
Node(char *str) {
s = new char[100];
strcpy(s, str);
}
/* 手动实现拷贝构造函数
Node(const Node &other) {
s = new char[100];
strcpy(s, other.s);
}
*/
};
int main() {
// freopen("in", "r", stdin);
Node a("hello");
Node b(a);
cout << b.s << endl;
strcpy(a.s, "bad copy");
cout << b.s << endl;
return 0;
}
正确的写法应该自己写一个拷贝函数,而不是用默认的,应该尽量的避免浅拷贝。
如何控制一个类只能在堆或栈上创建对象
在 C++ 中创建对象的方法有两种,一种是静态建立,一个是动态建立。
静态建立由编译器为对象分配内存,通过调用构造函数实现。这种方法创建的对象会在栈上。 静态建立由用户为对象分配内存,通过 new 来实现,间接调用构造函数。这种方法创建的对象会在堆上。
只能从堆上分配对象:
当建立的对象在栈上时,由编译器分配内存,因此会涉及到构造函数和析构函数。那么如果无法调用析构函数呢?也就是说析构函数是 private 的,编译器会先检查析构函数的访问性,由于无法访问,也就防止了静态建立。
但这种方法存在一种缺点,就是把析构函数设成 private 后,如果这个类要作为基类的话,析构函数应该设成虚函数,而设成 private 后子类就无法重写析构函数,所以应该把析构函数设成 protected。然后额外设置一个接口来 delete。
class Node {
public:
Node(){};
void Destroy() {
delete this;
}
protected:
~Node(){};
};
此时解决了静态建立的过程,但使用时,通过 new 创建对象,通过 Destroy 函数释放对象,为了统一,可以把构造函数和析构函数都设成 protected,重写函数完成构造和析构过程。
class Node {
public:
static Node* Create() {
return new Node();
}
void Destroy() {
delete this;
}
protected:
Node(){};
~Node(){};
};
只能从栈上分配对象:
同样的道理,只需要禁止通过动态建立对象就可以实现在栈上分配对象,所以可以重载 new 和 delete 并设为 private,使用户只能静态建立对象。
class Node {
public:
Node(){};
~Node(){};
private:
void* operator new(size_t t){}
void operator delete(void* p){}
};
memcpy 和 memmove 的实现
memcpy 可以直接通过指针自增赋值,但要求源地址和目的地址无重合。
void mymemmove1(void* s, const void* t, size_t n) {
char *ps = static_cast<char*>(s);
const char *pt = static_cast<const char*>(t);
while(n--) {
*ps++ = *pt++;
}
}
如果源地址和目的地址存在重合,会因为地址的重合导致数据被覆盖,所以要通过 memmove 来实现,需要从末尾往前自减赋值。
为了加快速度还可以使用 4 字节赋值的方式
// 直接按字节进行 copy
void mymemmove1(void* s, const void* t, size_t n) {
char *ps = static_cast<char*>(s);
const char *pt = static_cast<const char*>(t);
if(ps<=pt && pt<=ps+n-1) {
ps = ps+n-1;
pt = pt+n-1;
while(n--) {
*ps-- = *pt--;
}
} else {
while(n--) {
*ps++ = *pt++;
}
}
}
// 加快速度,每次按 4 字节进行 copy
void mymemmove2(void *s, const void *t, size_t n) {
int *ts = static_cast<int*>(s);
const int *tt = static_cast<const int*>(t);
char *ps = static_cast<char*>(s);
const char *pt = static_cast<const char*>(t);
int x = n/4, y = n%4;
if(ps<=pt && pt<=ps+n-1) {
ps = ps+n-1;
pt = pt+n-1;
while(y--) {
*ps-- = *pt--;
}
ps++, pt++;
ts = reinterpret_cast<int*>(ps);
tt = reinterpret_cast<const int*>(pt);
ts--, tt--;
while(x--) {
*ts-- = *tt--;
}
} else {
while(y--) {
*ps++ = *pt++;
}
ts = reinterpret_cast<int*>(ps);
tt = reinterpret_cast<const int*>(pt);
while(x--) {
*ts++ = *tt++;
}
}
}
通过重载 new/delete 来检测内存泄漏的简易实现
讲每次 new 产生的内存记录,并在 delete 的时候删去记录,那么最后剩下的就是发生内存泄漏的代码。
using namespace std;
class TraceNew {
public:
class TraceInfo {
private:
const char* file;
size_t line;
public:
TraceInfo();
TraceInfo(const char *File, size_t Line);
~TraceInfo();
const char* File() const;
size_t Line();
};
TraceNew();
~TraceNew();
void Add(void*, const char*, size_t);
void Remove(void*);
void Dump();
private:
map<void*, TraceInfo> mp;
} trace;
TraceNew::TraceInfo::TraceInfo() {
}
TraceNew::TraceInfo::TraceInfo(const char *File, size_t Line) : file(File), line(Line) {
}
TraceNew::TraceInfo::~TraceInfo() {
delete file;
}
const char* TraceNew::TraceInfo::File() const {
return file;
}
size_t TraceNew::TraceInfo::Line() {
return line;
}
TraceNew::TraceNew() {
mp.clear();
}
TraceNew::~TraceNew() {
Dump();
mp.clear();
}
void TraceNew::Add(void *p, const char *file, size_t line) {
mp[p] = TraceInfo(file, line);
}
void TraceNew::Remove(void *p) {
auto it = mp.find(p);
if(it != mp.end()) mp.erase(it);
}
void TraceNew::Dump() {
for(auto it : mp) {
cout << it.first << " " << "memory leak on file: " << it.second.File() << " line: " << it.second.Line() << endl;
}
}
void* operator new(size_t size, const char *file, size_t line) {
void* p = malloc(size);
trace.Add(p, file, line);
return p;
}
void* operator new[](size_t size, const char *file, size_t line) {
return operator new(size, file, line);
}
void operator delete(void *p) {
trace.Remove(p);
free(p);
}
void operator delete[](void *p) {
operator delete(p);
}
int main() {
int *p = new int;
int *q = new int[10];
return 0;
}
/*
0xa71850 memory leak on file: a.cpp line: 90
0xa719b8 memory leak on file: a.cpp line: 91
*/
垃圾回收机制
之前使用过,但现在不再使用或者没有任何指针再指向的内存空间就称为 "垃圾"。而将这些 "垃圾" 收集起来以便再次利用的机制,就被称为“垃圾回收”。
垃圾回收机制可以分为两大类:
基于引用计数的垃圾回收器 系统记录对象被引用的次数。当对象被引用的次数变为 0 时,该对象即可被视作 "垃圾" 而回收。但难以处理循环引用的情况。 基于跟踪处理的垃圾回收器 标记-清除:对所有存活对象进行一次全局遍历来确定哪些对象可以回收。从根出发遍历一遍找到所有可达对象(活对象),其它不可达的对象就是垃圾对象,可被回收。 标记-缩并:直接清除对象会造成大量的内存碎片,所以调整所有活的对象缩并到一起,所有垃圾缩并到一起,然后一次清除。 标记-拷贝:堆空间分为两个部分 From 和 To。刚开始系统只从 From 的堆空间里面分配内存,当 From 分配满的时候系统就开始垃圾回收:从 From 堆空间找出所有的活对象,拷贝到 To 的堆空间里。这样一来,From 的堆空间里面就全剩下垃圾了。而对象被拷贝到 To 里之后,在 To 里是紧凑排列的。接下来是需要将 From 和 To 交换一下角色,接着从新的 From 里面开始分配。
通过位运算实现加减乘除取模
加法操作
对于每一位而言,在不考虑进位的情况下,可以得到
0+0 = 0\\
0+1 = 1\\
1+0 = 1\\
1+1 = 0
显然,上面的情况符合 异或 操作且只有第四种情况发生了进位,进位情况符合 与 操作。在所有发生进位处,应该在更高的一位处加一,这个值可以通过 左移 操作实现。那么就可以得到
x+y = x \oplus y + (x \& y)<<1
可以发现,后面的式子变成了一个新的加法式,那么只要递归计算即可。当 (x & y)<<1 == 0 时,就消除了加法式。
ll add(ll x, ll y) {
ll newx = x^y;
ll newy = (x&y)<<1;
if(newy == 0) return newx;
return add(newx, newy);
}
减法操作
减法操作可以看成
同样可以通过加法式得到
ll sub(ll x, ll y) {
return add(x, add(~y, 1));
}
乘法操作
假设 y=1010,则可以关注于二进制上的 1 位,那么可以将 x*y 做出拆解
\begin{aligned}
xy &= x1010\
&= x1000 + x0010
\end{aligned}
而这个当乘数只有一个 1 时,可以通过二进制的左移操作实现。
ll mul(ll x, ll y) {
ll ans = 0;
int flag = x^y;
x = x<0 ? add(~x, 1) : x;
y = y<0 ? add(~y, 1) : y;
for(int i=0; (1ll<
if(y&(1ll<
ans = add(ans, x<
}
}
return flag<0 ? add(~ans, 1) : ans;
}
除法操作
和乘法操作思想一样,枚举答案每一位是否为 1,通过左移来得到乘积并减去。先从大的开始找,如果有一位是 1,那么就在答案将这一位设成 1。
ll divide(ll x, ll y) {
ll ans = 0;
int flag = x^y;
x = x<0 ? add(~x, 1) : x;
y = y<0 ? add(~y, 1) : y;
for(int i=31; i>=0; i--) {
if(x >= (1ll*y<
ans |= (1ll<
x = sub(x, 1ll*y<
}
}
return flag<0 ? add(~ans, 1) : ans;
}
取模操作
已经得到了除法的结果,那么取模操作也很容易实现了
ll mod(ll x, ll y) {
x = x<0 ? add(~x, 1) : x;
y = y<0 ? add(~y, 1) : y;
return sub(x, mul(y, divide(x, y)));
}
为什么子类对象可以赋值给父类对象而反过来却不行
子类继承于父类,它含有父类的部分,又做了扩充。如果子类对象赋值给父类变量,则使用该变量只能访问子类的父类部分。 如果反过来,这个子类变量如果去访问它的扩充成员变量,就会访问不到,造成内存越界。
为什么 free 时不需要传指针大小
free 要做的事是归还 malloc 申请的内存空间,而在 malloc 的时候已经记录了申请空间的大小,所以不需要传大小,直接传指针就可以。
手写链表实现 LRU
class LRU {
private:
struct Node {
int val;
Node *pre, *suf;
Node() {
pre = suf = nullptr;
}
Node(int _val) {
val = _val;
pre = suf = nullptr;
}
};
int size;
int capacity;
Node *head;
unordered_map<int, Node*> mp;
Node* find(int val) {
if(mp.count(val)) {
return mp[val];
} else {
return nullptr;
}
}
void del(Node *node) {
if(node == nullptr) return ;
node->pre->suf = node->suf;
node->suf->pre = node->pre;
mp.erase(node->val);
if(node == head) head = head->suf;
size--;
}
void add(Node *node) {
if(head == nullptr) {
head = node;
head->suf = head;
head->pre = head;
mp[node->val] = node;
} else {
node->suf = head;
node->pre = head->pre;
head->pre = node;
node->pre->suf = node;
mp[node->val] = node;
head = node;
}
size++;
}
public:
LRU() {
mp.clear();
head = nullptr;
size = capacity = 0;
}
void reverse(int _capacity) {
capacity = _capacity;
}
void insert(int val) {
Node *node = new Node(val);
Node *selectnode = find(val);
del(selectnode);
if(size == capacity) {
del(head->pre);
}
add(node);
}
void view() {
Node *p = head;
do {
cout << p->val << " ";
p = p->suf;
} while(p != head);
cout << endl;
}
}lru;
推荐阅读: