第一周
Algorithm:
-
215. 数组中的第K个最大元素 假设数组从小到大排序,则我们需要找到下表为n-k的元素。采用快排思想,在partition的时候,得到index,如果index=n-k则返回,小于n-k则在右区间递归找,否则左区间递归找
-
462. 最少移动次数使数组元素相等 II需要寻找中位数而不是平均数,寻找中位数最快的方法是什么?quick-select,即快排的思想。找到中位数后,求得每个元素到中位数的绝对值差之和。为什么是中位数,假设三个数,A<B<C,想要让ABC相等,调整成任意数D, A < D < C,则需要调整的次数为 D-A+C-D+|B-D|=C-A+|B-D|,因此最小的调整次数就是C-A,此时B=D,即调成中位数最好。
-
654. 最大二叉树 前序遍历,怎么处理根节点:找到最大值的下标,然后切分成子问题,分别构建左右子树
-
105. 从前序与中序遍历序列构造二叉树 前序遍历,如何处理根节点:在前序序列中取第一个找到根节点的值,然后在中序序列中找到值的下表,其左边即左子树,右边即右子树。前序序列的前半部分是左子树,后半部分是右子树
-
106. 从中序与后序遍历序列构造二叉树 前序遍历,如何处理根节点:在后序序列中取倒数第一个找到根节点的值,然后在中序序列中找到值的下表,其左边即左子树,右边即右子树。后序序列的前半部分是左子树,后半部分是右子树
Review:
robin hood hash
一种基于开放地址法的hash表,它主要的trick是:当你要插入一个元素时,如果当前probe到的元素比待插入的元素probe的次数少,那么就交换这两个元素。
robin hood hash 有以下特点:
- 装载因子可以很大,却对性能影响不大。例如0.9的装载因子都是完全可以的
- 相对于链表法,这种方式没有指针,cache命中率会更高
- 查询很快,并且性能方差很小
- 相对于其它的开放地址法,在寻找不存在的元素时,会更快。(因为它记录了hash表的最大探测长度,超过这个探测长度就停止探测。而其它开放地址法需要探测到空元素才能判定这个元素不存在)
Tip:
- 在非构造函数,非析构函数的成员函数中调用「虚函数」,也是多态!!!
//
class A {
public:
void func1(){
func2();
}
virtual void func2(){
cout<< "A 's func2 \n";
}
};
class B : public A{
public:
virtual void func2(){
cout << "B 's func2 \n";
}
}
int main(){
A * p = new B();
p->func1(); // 这里结果是调用B的func2!
}
- 改变一个对象的虚函数指针,指向父类,从而调用基类函数。这里主要知道对象的头8个字节就是它的虚函数指针就好。
// 还是上面的代码,只写main函数
int main(){
A a;
B b;
int * aptr = (int *) &a;
int * bptr = (int *) &b;
*bptr = *aptr;
A * basePtr = &b;
basePtr->func1(); // 这里结果是调用A的func2!
}
- c++ 11 类默认函数的控制
default
和delete
,分别显示采用默认函数或者禁用这个函数
class A{
public:
A()= default; // 采用默认构造函数
A(const A&) = delete;// 禁用copy 构造函数
private :
int c;
}