数据结构与算法之美(已面试部分)

2018-12-18  本文已影响18人  zhouluyao

1.题目:定义一个函数,输入一个链表的头结点,反转该链表并输出反转后链表的头结点。

//链表结点定义如下:
struct ListNode
{
    int m_nKey;
    ListNode *m_pNext;
};

算法实现:

ListNode * reverseList(ListNode *pHead)
{
    //防止链表断开,要保存当前节点,上一个节点,下一个节点
    //保存顺序:先保存上一个节点,再让当前节点指向上一个节点完成反转,遇到pNext==nullptr时到达链表尾部
    ListNode *pNode = pHead;
    ListNode *pPrev = nullptr;
    ListNode *pReverseHeadNode=nullptr;
    while (pNode!=nullptr)
    {
        //保存当前节点的下一个节点
        ListNode *pNext = pNode->m_pNext;

        //如果下一个节点为nullptr,找到了pReverseHeadNode
        if (pNode->m_pNext == nullptr)
            pReverseHeadNode = pNode;

        //反转
        pNode->m_pNext = pPrev;

        //把当前节点赋值为上一个节点
        pPrev = pNode;
        //把下一个节点赋值为当前节点
        pNode = pNext;

    }

    return pReverseHeadNode;

}

2.题目:输入两个链表,找出它们的第一个公共结点。

应用场景:

1.遍历两个view的第一个父控件

//链表结点定义如下:
struct ListNode
{
    int m_nKey;
    ListNode *m_pNext;
};

算法实现:

//找到两个链表的公共节点
ListNode *findCommonNode(ListNode *pFirstHead,ListNode *pLastHead)
{
//计算出两个链表的长度差值
    ListNode *pFirstNode = pFirstHead;
    ListNode *pLastNode = pLastHead;

    unsigned int firstListLength = getLinkedListLength(pFirstNode);
    unsigned int lastListLength = getLinkedListLength(pLastNode);
    unsigned int diff;
    if (firstListLength>lastListLength)
    {
        diff = firstListLength - lastListLength;
        //先让pFirstHead走diff个节点
        for (int i = 0; i < diff; i++)
        {
            pFirstNode = pFirstNode->m_pNext;
        }
    }
    else
    {
        diff = lastListLength - firstListLength;
        //先让pLastHead走diff个节点
        for (int i = 0; i < diff; i++)
        {
            pLastNode = pLastNode->m_pNext;
        }
    }
    while ((pFirstNode!=pLastNode)&&(pFirstNode!=nullptr) &&(pLastNode!=nullptr))
    {
        pFirstNode = pFirstNode->m_pNext;
        pLastNode = pLastNode->m_pNext;
                    
    }

    return pFirstNode;


}

unsigned int getLinkedListLength(ListNode *pHead)
{
    unsigned int length = 0;
    ListNode *pNode = pHead;

    while (pNode!=nullptr)
    {
        length++;
        pNode = pNode->m_pNext;
    }

    return length;
}

3.一个n阶的楼梯,每一步只能走1阶或者2阶,问n阶楼梯有多少种走法?

算法实现:

int f(int n)
{
    if (n == 1) return 1;
    if (n == 2) return 2;
    return f(n - 1) + f(n - 2);
}

4.题目:写一个函数,输入n,求斐波那契(Fibonacci)数列的第n项。斐波那契数列的定义如下:

f(0)=0;

f(1)=1;

f(n)=f(n-1)+f(n-2); //n>1

//降低时间复杂度,带数组缓存的实现
int fib(int n, int memo[] )
{

    if (n <= 1) return n;
    else if (!memo[n])
    {
        memo[n] = f(n - 1) + f(n - 2);
    }
    return memo[n];
}

5.链表中倒数第k个结点

题目:输入一个链表,输出该链表中倒数第k个结点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾结点是倒数第1个结点。例如一个链表有6个结点,从头结点开始它们的值依次是1、2、3、4、5、6。这个链表的倒数第3个结点是值为4的结点。

链表定义如下:

struct ListNode
{
    int m_nKey;
    ListNode *m_pNext;
};
ListNode *findKthToTail(ListNode *pHead,unsigned int k)
{
    //pHead==null,k=0,k超出链表的长度直接返回
    if(pHead==nullptr ||k==0) return nullptr;
    
    //先让前一个节点从链表头走k-1步,然后再让后一个节点同步走,前一个节点到达尾部,后一个节点就是倒数第k个节点
    
    ListNode *pAhead =pHead;
    ListNode *pBehind =nullptr;
    for (int i=0; i<k-1; i++) {
        if (pAhead->m_pNext!=nullptr) {
            pAhead =pAhead->m_pNext;
        }
        else
        {
            return nullptr; //保证k不超过链表的长度
        }
    }
    
    pBehind =pHead;
    while (pAhead->m_pNext!=nullptr) {
        pAhead =pAhead->m_pNext;
        pBehind=pBehind->m_pNext;
    }
    return pBehind;
    
}

6.合并两个排序的链表

题目:输入两个递增排序的链表,合并这两个链表并使新链表中的结点仍然是按照递增排序的。例如输入下图中的链表1和链表2,则合并之后的升序链表如链表3所示。

合并有序链表.jpg

链表结点定义如下:

struct ListNode
{
    int m_nKey;
    ListNode *m_pNext;
};

ListNode *Merge(ListNode *pHead1,ListNode *pHead2)
{
    if (pHead1==nullptr) return pHead2;
    if (pHead2==nullptr) return pHead1;
    //比较两个节点,哪个节点的值小,取出哪个节点,放到合并后链表的尾部
    ListNode *pMergeHead=nullptr;

    if(pHead1->m_nKey<pHead2->m_nKey)
    {
        pMergeHead =pHead1;
        pMergeHead->m_pNext=Merge(pHead1->m_pNext, pHead2);
    }
    else
    {
       pMergeHead =pHead2;
       pMergeHead->m_pNext=Merge(pHead1, pHead2->m_pNext);
    }

    return pMergeHead;
};

7.题目:判断一个链表是否包含环,如果包含环找出环的入口节点?

思路:

1.两个指针,快指针走2步,慢指针走一步,如果走到链表尾部还没相遇,说明链表中没有环,否则返回在链表中相遇的节点

2.找到相遇的节点,再次回到该节点时,计算环的大小loopLen

3.从链表的头部开始,一个指针先走LoopLen个节点,两个指针同时走,两个指针相遇的点即为环的入口点

truct ListNode
{
    int m_nKey;
    ListNode *m_pNext;
};

ListNode * MeetingNode(ListNode *pHead)
{
    if(pHead ==nullptr) return nullptr;
    
    ListNode *pSlow =pHead->m_pNext;
    if (pSlow==nullptr) return nullptr;
    
    ListNode *pFast =pSlow->m_pNext;
    
    while (pSlow!=nullptr &&pFast!=nullptr)
    {
        if (pSlow ==pFast) {
            return pSlow;
        }
        
        pSlow =pSlow->m_pNext;
        pFast=pFast->m_pNext;
        if (pFast!=nullptr) {
            pFast=pFast->m_pNext;
        }
    }
    
    return nullptr;
}


ListNode *EntryNodeOfLoop(ListNode *pHead)
{
    ListNode *meetingNode =MeetingNode(pHead); //第一次相遇的节点
    if(meetingNode==nullptr) return nullptr;
    
    //计算链表中环的大小
    int loopLen =1;
    ListNode *tempNode =meetingNode->m_pNext;
    while (meetingNode!=tempNode) {
        tempNode =tempNode->m_pNext;
        loopLen++;
    }
    
    //从头节点开始一个指针先走loopLen,两个指针一起走,第一次相遇的节点,即为环的入口
    ListNode *pNode1 =pHead;
    ListNode *pNode2 =pHead;
    for (int i=0; i<loopLen; i++) {
        pNode1 = pNode1->m_pNext;
    }
    
    while (pNode1!=nullptr &&pNode2!=nullptr)
    {
        if (pNode1==pNode2) {
            break;
        }
        pNode1=pNode1->m_pNext;
        pNode2=pNode2->m_pNext;
    }
    return pNode1;
}
  1. 给定二叉树,写出前序、中序、后序遍历的结果
    算法待实现:
waiting

9.不用递归,实现二叉树的前序、中序、后序遍历
算法待实现:

waiting

10.找到一个字符串最长无重复字符的子序列,你的算法复杂度是多少,什么是最大子序列问题,并求解,说出算法复杂度
算法待实现:

#include <string>
#include <map>

#define MAX( a , b) (a>b)?a:b;
int lengthOfLongestSubstring(std::string s)
{
    if (s.size() == 0) return 0;
    //利用map保存遍历的字符和下标
    std::map<char, int> c_map;
    int max = 0;
    for (int i = 0,j = 0; i < s.size(); i++)
    {
        //找到同样的字符时,更新下标
        if (c_map.find(s[i]) != c_map.end())
        {
            j = MAX(j, c_map[s[i]]+1);
        }
        c_map[s[i]] = i;
        max = MAX(max, i - j + 1);
    }
    //找到同样的字符时,计算这段子串的长度,下标i遍历,j标识子串起始位置
    return max;

    //把每个子串的长度记录下来,返回最大值
}
  1. 动态规划算法有哪些特性
    算法待实现:
waiting

12.一篇文章,如何准确的计算单词个数
算法待实现:

waiting

13.当我们想要查询 202.102.133.13 这个 IP 地址的归属地时,我们就在地址库中搜索,
发现这个 IP 地址落在 [202.102.133.0, 202.102.133.255] 这个地址范围内,
那我们就可以将这个 IP 地址范围对应的归属地“山东东营市”显示给用户了。
假设我们有 12 万条这样的 IP 区间与归属地的对应关系,如何快速定位出一个 IP 地址的归属地呢?

解题思路:ip可看做是32bit整数对应的一个整数
查找指定iP的归属地本质上就是查找最后一个小于等于给定值的元素的问题,查找对应iP取出对应归属地即可,如果没查到则提示没找到

//查找最后一个小于等于给定值的元素 :eg:2,4,5,7,8,9 找出最后一个小等于6的值
int binarySearch(int a[],int n,int value)
{
    int low =0;
    int high =n-1;

            while (low<=high)
            {
                 int mid =low+ ((high-low)>>1);
                 if(a[mid]>value)
                 {
                     high=mid-1;
                 }
                 else
                 {
                     if(mid==0 ||a[mid+1]>value) return a[mid];
                     low =mid+1;
                 }
            }
    return -1;

}
    int array[] ={2,4,5,7,8,9};
    int key = binarySearch(array,6,6);

题目14:快手算法面试,两个单向链表相加

比如:通过链表把两个数相加
//1->2->8->9
//3->8->1->6
//4->0->0->6->1

//实现思路:涉及到链表长度不相等处理,相加>9进位问题
1.在while循环里遍历2个链表对应节点值,取出值相加,到达尾部后值为0
2.只要链表没到尾部或者存在进位数,继续添加新节点,新节点值为(两个链表节点值+进位)%10,进位值为(两个链表节点值+进位)/10
3.初始化一个节点,指向新链表的头结点,然后遍历添加新值
ListNode *addSingleLinked(ListNode *phead1, ListNode *phead2)
{
    ListNode preHead(0);
    ListNode *p = &preHead;

    int extra = 0;
    while (phead1 || phead2 ||extra)
    {
        int newValue = (phead1?phead1->m_nKey:0 + phead2?phead2->m_nKey:0 + extra)%10;
        extra = (phead1 ? phead1->m_nKey : 0 + phead2 ? phead2->m_nKey : 0 + extra) / 10;
        ListNode *node=new ListNode(newValue);
        p->m_pNext = node;
        p = p->m_pNext;

        phead1 = (phead1 ? phead1->m_pNext : phead1);
        phead2 = (phead2 ? phead2->m_pNext : phead2);
    }
    return preHead.m_pNext;
}
上一篇 下一篇

猜你喜欢

热点阅读