首页 > 其他 > 详细

准备面试杂总

时间:2018-06-30 10:02:58      阅读:213      评论:0      收藏:0      [点我收藏+]

1.求整数序列中的中位数。

//首先想到的是先排序,后直接就求出来了

//参照这位大佬:https://blog.csdn.net/cpu_12593/article/details/48231213   有种恍然大悟的感觉

//洛谷1168题就是

Ⅰ。利用快排查找关键字,随机选取一个关键字进行划分,第一次划分的结果标记位置如果小于N/2,那么就将右半的部分再执行一次这个算法(从选关键字开始);如果关键字位置大于N/2,就将其左半部分在执行这个算法;直到关键字标记位置=N/2,才结束。这样的时间复杂度就是O(n),获取中位数时间为O(1).

Ⅱ。利用数据结构平衡二叉树(左右两颗子树的高度≤1,并且是递归定义的,左子树小于根节点,右子树相反。常用实现方法有红黑树,AVL树,treap树),每读取一个那么久插入到树中,最终根节点就是中位数,但是二叉树的插入调整算法比较复杂。时间复杂度为O(nlogn),获取时间复杂度是O(n),或者是通过查找根节点,用数组来表示的?

Ⅲ。参照这个:https://blog.csdn.net/yhf_2015/article/details/52315689   代码写的真厉害

利用堆来实现。左边是大顶堆,右边是小顶堆,小顶堆中元素永远要么和大顶堆中元素个数相同,要么比它多一个。也就是说将其平分为两部分。

当有元素插入时,如果两堆中元素个数相同,那么就尝试插入小顶堆中,插入的时候需要考虑,如果要插入的元素比小顶堆堆顶的元素小,那么就把大顶堆堆顶的元素插入到小顶堆中,并且将新元素插入的大顶堆中(都是为了保持数量关系。)。否则直接插入小顶堆。

如果两堆中元素个数不同(肯定是小顶堆中元素多了一个),那么就尝试加入大顶堆中,如果新元素比大顶堆堆顶元素大,那么就把小顶堆堆定元素挪过来,并且将新元素插入小顶堆。否则直接插入大顶堆。

//但是有没有这种情况就是把一个根去了之后,仍然仍然是不满足条件,那还接不接着挪点了?

技术分享图片
#include <bits/stdc++.h>
using namespace std;
const int maxn = 100010;
int a1[maxn];
int main(){
    int n;
    scanf("%d", &n);
    for(int i = 1; i <= n; i ++) scanf("%d", &a1[i]);
    
    priority_queue <int> q1;//优先队列默认是less,大顶堆。
    priority_queue <int, vector<int>, greater<int> > q2;//greater小顶堆
    int size1 = 0, size2 = 1;
    q2.push(a1[1]);//先放进
    
    for(int i = 2; i <= n+1; i ++){
        if(!(i&1))printf("%d\n", q2.top());//如果i是偶数的话,那么就打印小顶堆的堆定元素
//。比如说现在该查第4个数,那么前三个数中肯定是最小堆堆定元素是中位数。满足了题目要求。
        if(i > n) break;
        if(size2 > size1){
            if(a1[i] > q2.top()){
                q1.push(q2.top()), size1 ++;
                q2.pop();
                q2.push(a1[i]);
            }else q1.push(a1[i]), size1 ++;
        }else{
            if(a1[i] < q1.top()){
                q2.push(q1.top()), size2 ++;
                q1.pop();
                q1.push(a1[i]);
            }else q2.push(a1[i]), size2 ++;
        }
    }
    return 0;
}//太厉害了
贴大佬的代码

//真厉害,直接用优先队列就可以实现啊,没想到。。学习了。

2.c++中new和malloc的区别。

//真没考虑过这个问题。都是把数据分配在堆区。

参考这个:https://blog.csdn.net/linux_ever/article/details/50533149

Ⅰ。申请内存所在的位置。malloc是分配在堆中,new:自由存储区(那么可否为堆呢?要根据operator new的实现来,其中参数是一个指针,可以指向堆或静态区,在其处分配内存。或者甚至可以不分配内存)

Ⅱ。返回类型安全性。malloc返回是void*类型指针,需要进行强制类型转换;new返回的是严格匹配对象类型的指针,不用强制类型转换,是安全的。

Ⅲ。内存分配失败后返回值不同。malloc会返回NULL,而new则返回bac_alloc。

Ⅳ。是否需要指定内存大小。new不需要。malloc需要制定。

Ⅴ。是否调用析构函数。使用new/delete会调用对象的构造函数和析构函数。但是malloc不会。

//先写着几点吧,写多了也记不住。

3.堆和栈有什么区别?

//堆中是通过malloc分配的,栈是存储普通的类型变量,从低地址开始分配。其他的就像不知道了。

参考这位大佬:https://www.cnblogs.com/mysticCoder/p/4921724.html

栈的分配是由操作系统控制的,一般存放函数的局部变量等,使用完立即由操作系统释放。

堆是通过程序员手动申请和释放的,如果没有,则程序结束后由OS自动回收。

堆中内存分分配容量比栈中大,说来也是,一些大的高位数组,直接普通定义不行,可以用new那么就可以了。

碎片的问题:堆中会产生,但是栈是连续分配的不会产生,

生长方向:堆是向上生长即向着内存增加的方向;但是栈是相反的,是向着内存减小的方向。

分配效率:当然是栈高了。堆还需要调用一系列的算法来实现。

4.什么是随机变量?

//就是随机生成的变量,比如java中用random生成,可以给它种子。不过这是概率论的知识吗?emmm

简单的说就是随机事件的数量表现。包括离散性(伯努利概型)、连续性随机变量(指数随机变量)等。参考:百度百科随机变量定义。

 在一定条件下,并不一定出现相同结果的现象称谓为随机现象,其结果称为随机变量,比如任意时刻车站的人数。

5.如何反转一个单链表?

//头和尾调转,肯定是不行了,是单向的,对称调转的话那肯定就每次都需要遍历。。

参照这个:https://blog.csdn.net/hyqsong/article/details/49429859

 Ⅰ。从原链表的头部取元素插入到新链表的头部

Ⅱ。每次都把原来的头节点的后边一个元素放到最前。

 

准备面试杂总

原文:https://www.cnblogs.com/BlueBlueSea/p/9245948.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!