剑指offer刷题笔记

因为剑指offer的题目比较简单,所以就做成合集了,刷一题更新一题。

1 二位数组中的查找

在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。

这个就很简单了,从最左下方开始找起,如果现在的元素比目标大就向上走,现在的元素比目标小就往右走。

class Solution {
public:
    bool Find(int target, vector<vector<int> > array) {
        
        int i=array.size()-1;
        int j = 0;
        while( i >= 0 && j < array[0].size() ){
            int cur_ele = array[i][j];
            if( cur_ele == target ) return true;
            else if( cur_ele < target ) j += 1;
            else i -=1;
        }
        return false;
    }
};

2 替换空格

请实现一个函数,将一个字符串中的每个空格替换成“%20”。例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy。

这个其实也很简单,最要是我忘记了char* char a[],的用法,现在基本上都用string了,所以要复习一下。
原理就是找到空格的位置,然后从后向前添加。
注意'\0'问题,char*最后一位是'\0'这个是循环的终止条件。

class Solution {
public:
void replaceSpace(char *str,int length) {
        //遍历一边字符串找出空格的数量
        if(str==NULL||length<0)
            return ;
        int i=0;
        int oldnumber=0;//记录以前的长度
        int replacenumber=0;//记录空格的数量
        while(str[i]!='\0')
            {
               oldnumber++;
               if(str[i]==' ')
                   {
                     replacenumber++;
                   }
                  i++; 
            }
        int newlength=oldnumber+replacenumber*2;//插入后的长度
        if(newlength>length)//如果计算后的长度大于总长度就无法插入
            return ;
        int pOldlength=oldnumber; //注意不要减一因为隐藏个‘\0’也要算里
        int pNewlength=newlength;
        while(pOldlength>=0&&pNewlength>pOldlength)//放字符
            {
              if(str[pOldlength]==' ') //碰到空格就替换
                  {
                     str[pNewlength--]='0';
                     str[pNewlength--]='2';
                     str[pNewlength--]='%';
                     
                  }
               else //不是空格就把pOldlength指向的字符装入pNewlength指向的位置
               {
                    str[pNewlength--]=str[pOldlength];
                   
               }
             pOldlength--; //不管是if还是elsr都要把pOldlength前移
             
           }
        

}
};

3 从头到尾打印链表

输入一个链表,按链表值从尾到头的顺序返回一个ArrayList。

这个就更简单了,直接看代码吧。其实这个有很多解决方案,有时间再更新其他解。比如说用stack,递归打印等。

/**
*  struct ListNode {
*        int val;
*        struct ListNode *next;
*        ListNode(int x) :
*              val(x), next(NULL) {
*        }
*  };
*/
class Solution {
public:
    vector<int> printListFromTailToHead(ListNode* head) {
        vector<int> res;
        vector<int> res1;
        while(head != NULL){
            res.push_back( head -> val);
            head = head -> next;
        }
        // 反转
        for(int i=res.size() -1 ; i>=0; i--){
            res1.push_back(res[i]);
        }
        return res1;
    }
};

4 重建二叉树

输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回。

重建二叉树基本思想就是在前序遍历中找root,然后在中序遍历中找左右子树。

pre = {1,2,4,7,3,5,6,8}
vin = {4,7,2,1,5,3,8,6}
第一次迭代:从pre中找起,1为根节点,然后在vin中找,1左边的都是它左子树的树,1右边的都是它右子树的树。
第二次迭代:这时候vin变为{4,7,2}从pre中找起,2为根节点,然后在vin中找,只有左子树,右子树返回NULL。
第三次迭代....                            
/**
 * Definition for binary tree
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    TreeNode* reConstructBinaryTree(vector<int> pre,vector<int> vin) {
        if( pre.empty() || vin.empty() ) return NULL;
        else{
            // 构造函数
            TreeNode* root = new TreeNode(pre[0]);
            int pos_root = 0;
            for(int i=0; i<vin.size(); i++){
                if( vin[i] == pre[0] ) {pos_root = i; break;}
            }
            // 分割二叉树的两个数组
            vector<int> left_pre, right_pre, left_vin, right_vin;
            for(int i=0; i<pos_root; i++){
                left_pre.push_back(pre[i + 1]);
                left_vin.push_back(vin[i]);
            }
            for( int i=pos_root + 1; i<vin.size(); i++){
                right_pre.push_back(pre[i]);
                right_vin.push_back(vin[i]);
            }
            root->left = reConstructBinaryTree(left_pre, left_vin);
            root->right = reConstructBinaryTree(right_pre, right_vin);
            return root;
        }
    }
};

5 用两个栈实现队列

用两个栈来实现一个队列,完成队列的Push和Pop操作。 队列中的元素为int类型。

stack性质是先进后出,队列性质是先进先出。原理也很简单,两次先进后出不就是先进先出了。。。

class Solution
{
public:
    void push(int node) {
        stack1.push(node);
    }
 
    int pop() {
        if( stack2.empty() ){
            while( !stack1.empty() ){
                stack2.push(stack1.top());
                stack1.pop();
            }
        }
        int res = stack2.top();
        stack2.pop();
        return res;
    }
 
private:
    stack<int> stack1;
    stack<int> stack2;
};

6 旋转数组的最小数字

把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素。 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。 NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。

这道题有点似曾相识的感觉,貌似在Leetcode里做过。用暴力的方法目测会超时,然后一般这种数组题目就是用分治法或者二分法做一下的样子。
先尝试二分法,二分法最主要就是找到二分的条件。

Array = { 3,4,5,6,7,8,9,10,1,2,3}
mid = ( left + right ) / 2
(1) mid > right, min在mid+1到right之间。
(2) mid < left, min在left到mid-1之间。

还需要注意一种情况,mid == right == left,因为无法判断最小值到底在哪一部分,所以只能暴力查找了。
要单独拿出来判断。
eg {0, 1, 1, 1, 1} 的两个旋转 {1,0, 1, 1,1} {1, 1, 1, 0, 1}
class Solution {
public:
int minNumberInRotateArray(vector<int> rotateArray) {
        if (rotateArray.empty()) return 0;
        int left = 0, right = rotateArray.size() - 1;
        while (left < right) {
            //确认子数组是否是类似1,1,2,4,5,..,7的非递减数组
            if (rotateArray[left] < rotateArray[right]) return rotateArray[left];
             
            int mid = left + (right - left) / 2;
            //如果左半数组为有序数组
            if (rotateArray[left] < rotateArray[mid])
                left = mid + 1;
            //如果右半数组为有序数组
            else if (rotateArray[mid] < rotateArray[right])
                right = mid;
            //否则,rotateArray[left] == rotateArray[mid] == rotateArray[right]
            else {
                ++left;
            }
        }
        return rotateArray[left];
    }
};

7 斐波那契数列

大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0)。
n<=39

递推式是 F0=0,F1=1,Fn=Fn-1+Fn-2(n>=2,n∈N*)
直接递推会超时,所以采用了迭代求解的方法。

class Solution {
    int helper(int product, int product_pre, const int& n, int cur_n){
        if( n == 0 ) return 0;
        if( n == 1 ) return 1;
        int temp_product = product + product_pre;
        if( cur_n == 1 ) temp_product = 1;
        if( cur_n == n ){
          return temp_product;   
        }
        if( cur_n == 0 ) return helper(0, 0, n, cur_n + 1);
        else return helper(temp_product, product, n, cur_n + 1);
    }
public:
    int Fibonacci(int n) {
        return helper(0, 0, n, 0);
    }
};

8 跳台阶

一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果)。

问题可以从每个台阶n上来看,对于n来说,它只能够由n-2这个台阶跳两步到达,或者n-1这个台阶跳1步到达。也就说n这个台阶的种类取决于n-2这个台阶和n-1这个台阶的种类。
F(n)=F(n-1)+F(n-2)

class Solution {
public:
    int jumpFloor(int number) {
        if( number == 1 ) return 1;
        if( number == 2 ) return 2;
        return jumpFloor(number - 1) + jumpFloor(number - 2);
    }
};

这个和上一题不是一样一样的么,为什么这道题直接用式子递推就不会超时?测试用例需要扩充。

9 变态跳台阶

一只青蛙一次可以跳上1级台阶,也可以跳上2级……它也可以跳上n级。求该青蛙跳上一个n级的台阶总共有多少种跳法。

思路和上一题一模一样的。
F(n) = F(n-1)+F(n-2)+F(n-3)+...+F(1)
F(n-1) = F(n-2)+F(n-3)+F(n-4)+..+F(1)
下面的式子带入上面就有:
F(n)=2F(n-1)

class Solution {
public:
    int jumpFloorII(int number) {
        int n = number;
        if( n == 1 ) return 1;
        return 2 * jumpFloorII( n -1 );
    }
};

10 矩形覆盖

我们可以用2*1的小矩形横着或者竖着去覆盖更大的矩形。请问用n个2*1的小矩形无重叠地覆盖一个2*n的大矩形,总共有多少种方法?

其实我想自己画图的,但是没有找到合适的工具。
所以这里就盗图一下:

递推规律图解.png

就是说新加了一行之后,新的一行只有横着排一行或者竖着排两行两种排列形式。也就对应着F(n-1)和F(n-2)。所以问题又回到了斐波切纳数列上了。

class Solution {
public:
    int rectCover(int number) {
        if(number == 0 )return 0;
        if(number == 1) return 1;
        if(number == 2) return 2;
        return (rectCover(number - 1) + rectCover(number - 2));
    }
};

11 二进制中1的个数

输入一个整数,输出该数二进制表示中1的个数。其中负数用补码表示。

首先判断n是不是负数,当n为负数的时候,直接用后面的while循环会导致死循环,因为负数
向左移位的话最高位补1! 因此需要一点点特殊操作,可以将最高位的符号位1变成0,也就
n &0x7FFFFFFF,这样就把负数转化成正数了,唯一差别就是最高位由1变成0,因为少了
一个1,所以count加1。之后再按照while循环里处理正数的方法来操作就可以啦!

class Solution {
public:
     int  NumberOf1(int n) {
         // 处理负数
         int counter = 0;
         if( n < 0 ){
             n &=  0x7FFFFFFF;
             counter ++;
         }
         while( n != 0 ){
             counter += n & 1;
             n = n >> 1;
         }
         return counter;
     }
};

12数值的整数次方

给定一个double类型的浮点数base和int类型的整数exponent。求base的exponent次方。

这里我偷懒了,写了个很简单的解,最后也通过了。这个系统令人匪夷所思。这个问题应该是用递归求解的,在leetcode里遇到过。

class Solution {
public:
    double Power(double base, int exponent) {
        double result = 1;
        if( exponent < 0 ){
            base = 1 / base;
            exponent = abs(exponent); 
        }
        for( int i = 0; i< exponent; i++){
            result *= base;
        }
        return result;
    }
};

13调整数组顺序使奇数位于偶数前面

输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变。

这道题思想应该和插排差不多,但是我随便写了一个简单解,然后就通过了,于是我也没深究。

class Solution {
public:
    void reOrderArray(vector<int> &array) {
        vector<int> rec1;
        vector<int> rec2;
        for( auto i : array){
            if( i % 2 == 0 ) rec2.push_back(i);
            else rec1.push_back(i);
        }
        for( int i =0; i< array.size() ;i ++){
            if( i < rec1.size() ) array[i] = rec1[i];
            else array[i] = rec2[i-rec1.size()];
        }
        
    }
};

14 链表中倒数第k个节点

输入一个链表,输出该链表中倒数第k个结点。

典型的双指针问题,申明两个指针p1,p2,让p2先走k步,然后p1,p2一块走,p2走到末尾时候,p1指向的位置就是我们要求的解。

/*
struct ListNode {
    int val;
    struct ListNode *next;
    ListNode(int x) :
            val(x), next(NULL) {
    }
};*/
class Solution {
public:
    ListNode* FindKthToTail(ListNode* pListHead, unsigned int k) {
if(pListHead==NULL||k==0)
            return NULL;
        ListNode*pTail=pListHead,*pHead=pListHead;
        for(int i=1;i<k;++i)
        {
            if(pHead->next!=NULL)
                pHead=pHead->next;
            else
                return NULL;
        }
        while(pHead->next!=NULL)
        {
            pHead=pHead->next;
            pTail=pTail->next;
        }
        return pTail;
    }
};
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,902评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,037评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,978评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,867评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,763评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,104评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,565评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,236评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,379评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,313评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,363评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,034评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,637评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,719评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,952评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,371评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,948评论 2 341

推荐阅读更多精彩内容

  • 要求:不仅仅能实现相应的功能,还需要保证代码的鲁棒性,并且能够分析代码的空间复杂度和时间复杂度。 二维数组中的查找...
    二十四桥客_阅读 963评论 0 1
  • 本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那...
    波波波先森阅读 4,064评论 0 23
  • 前言 2. 实现 Singleton 3. 数组中重复的数字 4. 二维数组中的查找 5. 替换空格 6. 从尾到...
    Observer_____阅读 2,893评论 0 1
  • 这两日,我看了本《阿凡提笑话合集》,几百个笑话时而让我微微一笑,时而让我沉思不语,时而让我捧腹大笑,时而让我...
    周言欣文阅读 354评论 0 0
  • 今天是我和老爸的生日。 二十四周岁,这之前的青春是热烈的,感性的,纯粹的。所作所为似乎都可以任性而为,却也对于理想...
    敬一不二阅读 311评论 0 0