day1-牛客67道剑指offer-JZ4 JZ6 JZ7 JZ9 JZ11 JZ69 JZ70 替换空格 斐波那契数列及其变形 左移/右移运算符

文章目录

  • 1. JZ4 二维数组中的查找
    • 暴力法
    • 右上角往左下角逼近
    • 二分查找-左闭右开区间
  • 2. 替换空格
  • 3. JZ6 从尾到头打印链表
  • 4. JZ7 重建二叉树
    • 思路1
    • 哈希加速
  • 5. JZ9 用两个栈实现队列
  • 6. JZ11 旋转数组的最小数字
    • 常规遍历
    • 二分法
  • 7. 斐波那契数列
    • 动态规划
    • 递归
  • 8. JZ69 跳台阶
    • 动态规划
    • 递归
  • 9. JZ71 跳台阶扩展问题
    • 动态规划-看题解
    • 动态规划-优化空间
    • 数学规律-优化时间空间-左移运算
  • 10. JZ70 矩形覆盖
    • 动态规划 数组写法
    • 动态规划 三个变量
  • 补充内容:左移运算符与右移运算符

1. JZ4 二维数组中的查找

在这里插入图片描述

暴力法

两层for循环,自测可通过,超时,时间复杂度m*n

右上角往左下角逼近

利用递增,右上角为分割点

    bool Find(int target, vector<vector<int> >& array) {       
        //右上角逐渐逼近左下角 递增
        if(array.empty() || array[0].empty()) return false;
        int row = array[0].size();
        int col = array.size();
        int w = row - 1, h = 0;
        while(w >= 0 && h < col)
        {
            if(array[h][w] > target) w--;//array[h][w]表示右上角 target在左边 往左走
            else if(array[h][w] < target) h++;//target在下面 往下走
            else return true;
        }
        return false;
    }

二分查找-左闭右开区间

也可以是左闭右闭区间实现

    bool Find(int target, vector<vector<int> >& array) {
        //二分查找
        if(array.empty() || array[0].empty()) return false;
        
        //遍历每一行 二分
        for(int i=0; i<array.size(); i++)
        {
            if(binaryfind(array[i], target)) return true;
        }
        return false;
    }
     bool binaryfind(vector<int>nums, int target)
    {
        //左闭右开区间
        int start = 0, end = nums.size();
        int mid = 0;
        for(int i=0; i<nums.size(); i++)
        {
            mid = start + (end - start) / 2;
            if(target < nums[i]) end = mid;
            else if(target > nums[i]) start = mid + 1;
            else return true;
        }
        return false;
    }

2. 替换空格

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

统计空格长度,倒序遍历替换
第一次写的时候用的是for(int i=length; i>=0 && i<newlen; i--)条件
这次写的时候用的是for(int i=length; i>=0 && newlen!=i; i--)条件

class Solution {
public:
	void replaceSpace(char *str,int length) {
		int spacelen = 0;
		for(int i=0; i<length; i++)
		{
			if(str[i] == ' ') spacelen++;
		}
		int newlen = length + 2*spacelen;
		//替换 如果newlen=i 说明此时前面已经都没有空格了,可以节约一部分时间,而不是一直赋值下去
		for(int i=length; i>=0 && newlen!=i; i--)
		//for(int i=length; i>=0 && i<newlen; i--)
		{
			if(str[i] == ' ')
			{
				str[newlen--] = '0';
				str[newlen--] = '2';
				str[newlen--] = '%';
			}
			else str[newlen--] = str[i];
		}

		int spaceCount = 0;
	}
};

3. JZ6 从尾到头打印链表

题目:输入一个链表,按链表从尾到头的顺序返回一个ArrayList

正序保存,然后reverse返回。或者返回逆序

/**
*  struct ListNode {
*        int val;
*        struct ListNode *next;
*        ListNode(int x) :
*              val(x), next(NULL) {
*        }
*  };
*/
#include <algorithm>
class Solution {
public:
    vector<int> printListFromTailToHead(ListNode* head) {
        if(head == NULL) return vector<int>();
        //正序保存+reverse
        vector<int> result;
        ListNode* cur = head;
        while(cur)
        {
            result.push_back(cur->val);
            cur = cur->next;
        }
        /*
        reverse(result.begin(), result.end());
        return result;
        */
        
        return vector<int>(result.rbegin(),result.rend());
    }
};

4. JZ7 重建二叉树

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

力扣上有一道题是根据中序和后序遍历结果重建树,106. 从中序与后序遍历序列构造二叉树,这道题是根据中序和前序遍历结果重建树。

思路1

根节点分割中序遍历结果,左子树节点数分割前序遍历结果

/**
 * struct TreeNode {
 *	int val;
 *	struct TreeNode *left;
 *	struct TreeNode *right;
 *	TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 * };
 */
#include <iterator>
class Solution {
public:
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 
     * @param preOrder int整型vector 
     * @param vinOrder int整型vector 
     * @return TreeNode类
     */
    TreeNode* reConstructBinaryTree(vector<int>& preOrder, vector<int>& vinOrder) {
        if (preOrder.size() == 0 || vinOrder.size() == 0) {
            return NULL;
        }
        //distance两个迭代器之间元素个数 find查找两个迭代器之间是否有元素preOrder[0]
        int rootindex = distance(vinOrder.begin(), find(vinOrder.begin(), vinOrder.end(), preOrder[0]));
        //前序遍历 左闭右开区间 左子树 右子树分割 利用节点个数 注意去掉根节点
        vector<int> left_pre(preOrder.begin() + 1, preOrder.begin() + 1 + rootindex);
        vector<int> right_pre(preOrder.begin() + 1 + rootindex, preOrder.end());
        //中序遍历 左闭右开区间 左子树 右子树分割 利用根节点
        vector<int> left_vin(vinOrder.begin(), vinOrder.begin() + rootindex);
        vector<int> right_vin(vinOrder.begin() + rootindex + 1, vinOrder.end());

        //递归构造
        TreeNode* head = new TreeNode(preOrder[0]);
        head->left = reConstructBinaryTree(left_pre, left_vin);
        head->right = reConstructBinaryTree(right_pre, right_vin);
        return head;
    }
};

哈希加速

/**
 * struct TreeNode {
 *	int val;
 *	struct TreeNode *left;
 *	struct TreeNode *right;
 *	TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 * };
 */
class Solution {
public:
    /**
     * 代码中的类名、方法名、参数名已经指定,请勿修改,直接返回方法规定的值即可
     *
     * 
     * @param preOrder int整型vector 
     * @param vinOrder int整型vector 
     * @return TreeNode类
     */
    TreeNode* reConstructBinaryTree(vector<int>& preOrder, vector<int>& vinOrder) {
        
        //哈希加速
        unordered_map<int, int> hashmap;
        for(int i=0; i<vinOrder.size(); i++)
        {
            hashmap.insert(make_pair(vinOrder[i], i));
        }
        //递归
        return tralversal(hashmap, preOrder, 0, vinOrder, 0, vinOrder.size()-1);
    }

    TreeNode* tralversal(unordered_map<int, int>& hashmap, vector<int>& pre, int start_pre, vector<int>& vin, int start_vin, int end_vin)
    {
        //可以取等号
        if(start_pre > pre.size() || start_vin > end_vin) return nullptr;
        TreeNode* root = new TreeNode(pre[start_pre]);
        int rootindex = hashmap[pre[start_pre]];
        //第三个参数是 前序遍历中左子树的 头结点 索引位置 
        //第五、六参数是索引范围是中序遍历的左子树区间 start_vin~rootindex - 1 左闭右闭
        root->left = tralversal(hashmap, pre, start_pre+1, vin, start_vin, rootindex - 1);
        //第三个参数是 前序遍历中右子树的 头结点 索引位置 在前序遍历中是start_pre+1+左子树节点数
        //第五、六参数是索引范围是中序遍历的右子树区间 rootindex+1~end_vin 左闭右闭
        root->right = tralversal(hashmap, pre, start_pre+1+rootindex-start_vin, vin, rootindex+1, end_vin);
        return root;
    }
};

5. JZ9 用两个栈实现队列

题目描述:完成队列的Push和Pop操作。 队列中的元素为int类型

两个栈实现,一个保存元素,一个辅助

class Solution
{
public:
    void push(int node) {
        stack1.push(node);
    }

    int pop() {
        //队列是双头 先入先出;栈是单头 先进后出 所以要倒序一下
        while(stack1.size() != 1)//保留栈底元素
        {
            stack2.push(stack1.top());
            stack1.pop();
        }
        int target = stack1.top();
        stack1.pop();
        //剩下的元素再放回去stack1中
        while (!stack2.empty()) {
            stack1.push(stack2.top());
            stack2.pop();
        }
        return target;
    }

private:
    stack<int> stack1;//保存元素
    stack<int> stack2;//辅助栈
};

6. JZ11 旋转数组的最小数字

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

常规遍历

class Solution {
public:
    int minNumberInRotateArray(vector<int>& nums) {
        if(nums.size() == 0) return 0;
        //遍历整个数组
        for(int i=0; i<nums.size(); i++)
        {
            if(nums[i] > nums[i+1]) return nums[i+1];
        }
        return nums[0];
    }
};

二分法

class Solution {
public:
    int minNumberInRotateArray(vector<int>& nums) {
        if(nums.size() == 0) return 0;
        //二分法 左闭右开 最值在两头
        int left = 0, right = nums.size()-1;
        while(left+1 < right)
        {
            int mid = left + (right - left) / 2;
            if(nums[mid] < nums[right]) right = mid;//说明右边有序,那就向左边走
            else if(nums[mid] == nums[right]) right = right-1;
            else left = mid;
        }
        return min(nums[left], nums[right]);
    }
};

7. 斐波那契数列

题目描述:大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0,第1项是1)。n≤39

动态规划

三个数值保存在一个数组,注意最后返回值是n-1对3取余数

class Solution {
public:
    int Fibonacci(int n) {
        if(n == 1 || n == 2) return 1;
        if(n == 3) return 2;
        vector<int> F(3);
        F[0] = 1;
        F[1] = 1;
        F[2] = 2;
        for(int i=3; i<n; i++)
        {
            F[i % 3] = F[(i-1) % 3] + F[(i-2) % 3];
        }
        return F[(n-1) % 3];
    }
};

递归

class Solution {
public:
    int Fibonacci(int n) {
        //递归
        if(n == 0) return 0;
        if(n == 1 || n == 2) return 1;
        return Fibonacci(n-1) + Fibonacci(n-2);
    }
};

8. JZ69 跳台阶

题目描述:一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果)。
相当于斐波那契数列

动态规划

动态规划的定义是跳n级台阶有多少种跳法

class Solution {
public:
    int jumpFloor(int number) {
        //动态规划
        if(number == 1) return 1;
        if(number == 2) return 2; 
        vector<int> dp(3);
        dp[0] = 1;
        dp[1] = 2;
        for(int i=2; i<number; i++)
        {
            dp[i % 2] = dp[(i-1) % 2] + dp[(i-2) % 2];
        }
        return dp[(number-1) % 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. JZ71 跳台阶扩展问题

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

动态规划-看题解

时间复杂度: O ( n ) O(n) O(n),其中n为台阶数,一次遍历;空间复杂度: O ( n ) O(n) O(n),辅助数组dp的长度为n

在这里插入图片描述在这里插入图片描述

class Solution {
public:
    int jumpFloorII(int number) {
        //动态规划-题解
        vector<int> dp(number+1);
        dp[0] = 1;
        dp[1] = 1;
        for(int i=2; i<=number; i++) dp[i] = 2 * dp[i-1];
        return dp[number];
    }
};

动态规划-优化空间

在上面看了题解写之后,优化了空间,时间复杂度: O ( n ) O(n) O(n);空间复杂度: O ( 1 ) O(1) O(1)

class Solution {
public:
    int jumpFloorII(int number) {
        //动态规划-空间优化
        vector<int> dp(3);
        dp[0] = 1;
        dp[1] = 1;
        for(int i=2; i<=number; i++)
        {
            dp[i % 2] = 2 * dp[(i-1) % 2];
        }
        return dp[number % 2];
    }
};

数学规律-优化时间空间-左移运算

在这里插入图片描述
在这里插入图片描述
时间复杂度: O ( 1 ) O(1) O(1),一次位运算;空间复杂度: O ( 1 ) O(1) O(1)

通过左移运算求幂次方来优化时间复杂度,也就是说pow(2, number - 1) 等价于 1<<number-11<<number-1就是将1左移number-1位

class Solution {
public:
    int jumpFloorII(int number) {
        //数学规律-时间优化
        if(number <= 1) return 1;
        //return pow(2, number - 1);//计算2的number-1次方 时间复杂度是n 空间是1
        return 1<<number-1;//将1左移number-1位 通过左移运算求幂次方 时间/空间复杂度是1
    }
};

10. JZ70 矩形覆盖

题目描述:我们可以用21的小矩形横着或者竖着去覆盖更大的矩形。请问用n个21的小矩形无重叠地覆盖一个2n的大矩形,总共有多少种方法?注意:约定 n == 0 时,输出 0
比如n=3时,2
3的矩形块有3种覆盖方法,如下

宽度永远是2,那么题目换个说法就是,用n个边长为1的短线可以变成边长为n的长线,总共有多少种方法?也就是n个1变成n,一共有多少种办法?其实就是斐波那契额数列。

动态规划 数组写法

class Solution {
public:
    int rectCover(int number) {
        if(number <= 2) return number;
        vector<int> dp(3);
        dp[0] = 0;
        dp[1] = 1;
        dp[2] = 2;
        for(int i=3; i<=number; i++)
        {
            dp[i % 3] = dp[(i-1) % 3] + dp[(i-2) % 3];
        }
        return dp[number % 3];
    }
};

动态规划 三个变量

class Solution {
public:
    int rectCover(int number) {
        if(number <= 2) return number;
        //动态规划 三个变量
        int first = 1, second = 2, third = 0;
        for(int i=3; i<=number; i++)
        {
            third = first + second;
            //下一轮计算前要更新
            first = second;
            second = third;
        }
        return third;
    }
};

补充内容:左移运算符与右移运算符

  1. <<左移运算符,逻辑移位:expr1 << expr2,表示 expr1 左移 expr2 位,数值上表示 expr1 扩大了 2^expr2 倍;
  2. >>右移运算符,算术移位:expr1>>expr2,表示 expr2 右移 expr2 位,数值上表示 expr1 缩小了 2^expr2 倍;
  3. 左边的数表示被移位的数字,1<<n = 2^n,n<<1 = 2*n。左移就是扩大2的移位数次方,右移就是缩小2的移位数次方

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/58608.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

PS - Photoshop 实现涂抹功能 (橡皮擦、图章、吸管、画笔)

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/131997323 在 Photoshop 中&#xff0c;橡皮擦工具&#xff0c;以及吸管工具和画笔工具可以配合使用&#xff0c;实现涂抹功能&#xff0c;再通过…

AMEYA360:瑞萨电子MCU和MPU产品线将支持Microsoft Visual Studio Code

全球半导体解决方案供应商瑞萨电子宣布其客户现可以使用Microsoft Visual Studio Code&#xff08;VS Code&#xff09;开发瑞萨全系列微控制器&#xff08;MCU&#xff09;和微处理器&#xff08;MPU&#xff09;。瑞萨已为其所有嵌入式处理器开发了工具扩展&#xff0c;并将其…

zookeeper入门学习

zookeeper入门学习 zookeeper应用场景 分布式协调组件 客户端第一次请求发给服务器2&#xff0c;将flag值修改为false&#xff0c;第二次请求被负载均衡到服务器1&#xff0c;访问到的flag也会是false 一旦有节点发生改变&#xff0c;就会通知所有监听方改变自己的值&#…

ConcurrentHashMap 的简单介绍

ConcurrentHashMap是Java集合框架中的一个并发容器&#xff0c;它是线程安全的哈希表的实现。它被设计为比Hashtable和SynchronizedMap&#xff08;通过使用同步方法或块来保证线程安全&#xff09;更高效和可扩展的替代品。 ConcurrentHashMap具有以下特点&#xff1a; 线程…

Docker安装RabbitMQ镜像

步骤1&#xff1a;拉取镜像 docker pull rabbitmq:management 步骤2&#xff1a;运行 docker run -d –-name rabbit -e RABBITMQ_DEFAULT_USERadmin -e RABBITMQ_DEFAULT_PASSadmin -p 15672:15672 -p 5672:5672 -p 25672:25672 -p 61613:61613 -p 1883:1883 rabbitmq:mana…

Windows同时安装两个版本的JDK并随时切换,以JDK6和JDK8为例,并解决相关存在的问题(亲测有效)

Windows同时安装两个版本的JDK并随时切换&#xff0c;以JDK6和JDK8为例&#xff0c;并解决相关存在的问题&#xff08;亲测有效&#xff09; 1.下载不同版本JDK 这里给出JDK6和JDK的百度网盘地址&#xff0c;具体安装过程&#xff0c;傻瓜式安装即可。 链接&#xff1a;http…

Redis学习总结

Redis学习总结 文章目录 Redis学习总结Radis基本介绍docker的安装基本数据结构通用命令字符型key的层次结构Hash类型Listset sortedset集合redis的java客户端jedis的使用jedis连接池的配置 SpringDataRedis自定义redistemplate的序列化与反序列化方式stringtemplate的使用 redi…

windows创建占用特定端口程序

默认情况下&#xff0c;远程桌面使用的是3389端口。如果您想将远程桌面端口更改为8005&#xff0c;以达到模拟程序占用端口8005的情况&#xff0c;可以执行以下操作&#xff1a; 如执行以下命令&#xff0c;则1&#xff0c;2&#xff0c;3步相同操作可以跳过&#xff0c;直接往…

【Java】Springboot脚手架生成初始化项目代码

Springboot配置生成初始化项目代码可以通过mvn的mvn archetype:generate 和阿里云原生应用脚手架&#xff08;地址&#xff09;、spring官方提供的start初始化生成页面(地址&#xff09;。 1、mvn archetype:generate 通过mvn选择对应的脚手架可以快速生成初始化代码&#xf…

一次有趣的Webshell分析经历

一次有趣的Webshell分析经历 1.拉取源代码2.解密后门代码3.分析webshell逻辑4.分析404的原因5.附&#xff1a;格式化后的php代码 1.拉取源代码 在对某目标做敏感目录收集时发现对方网站备份源代码在根目录下的 backup.tar.gz&#xff0c;遂下载&#xff0c;先使用D盾分析有没有…

shiro快速入门

文章目录 权限管理什么是权限管理&#xff1f;什么是身份认证&#xff1f;什么是授权&#xff1f; 什么是shiro&#xff1f;shiro的核心架构shiro中的三个核心组件 shiro中的认证shiro中的授权shiro使用默认Ehcache实现缓存shiro使用redis作为缓存实现 权限管理 什么是权限管理…

31.利用linprog 解决 投资问题(matlab程序)

1.简述 语法&#xff1a;[X,FVAL] linprog(f,a,b,Aeq,Beq,LB,UB,X0)&#xff1b; X 为最终解 &#xff0c; FVAL为最终解对应的函数值 *注意&#xff1a;求最大值时&#xff0c;结果FVAL需要取反* f 为决策函数的系数矩阵。 *注意&#xff1a;当所求为最大值…

SpringBoot实现数据库读写分离

SpringBoot实现数据库读写分离 参考博客https://blog.csdn.net/qq_31708899/article/details/121577253 实现原理&#xff1a;翻看AbstractRoutingDataSource源码我们可以看到其中的targetDataSource可以维护一组目标数据源(采用map数据结构)&#xff0c;并且做了路由key与目标…

LeetCode 26 题:删除有序数组的重复项

思路 在写这一个题时&#xff0c;我突然想到了Python中的 set&#xff08;&#xff09;函数可能会有大用处&#xff0c;便选择了用Python写。 set&#xff08;&#xff09;函数可以将列表转化为集合&#xff0c;集合会保证元素的单一性&#xff0c;所以会自动删去相同字符。 …

sqlserver命令插入另一个数据库的数据主键自增

1、数据情况 两个数据库字段是一致的&#xff0c;其中OBJECTID是主键字段&#xff0c;而且两个表都是从1自增排序 2、需求 现在需要将另一个数据库中的数据&#xff0c;通过sqlserver语句的方法&#xff0c;来插入数据&#xff0c;保持自增字段是自增的 解决方法 sqlserv…

基于边缘无线协同感知的低功耗物联网LPIOT技术:赋能智慧园区方案以及数字工厂领域

回到2000年左右&#xff0c;物联网的底层技术支撑还是“ZigBee”&#xff0c;虽然当时ZigBee的终端功耗指标其实也并不庞大&#xff0c;但是&#xff0c;“拓扑复杂导致工程实施难度大”、“网络规模小导致的整体效率低下”都成为限制其发展的主要因素。 LPWAN&#xff0c;新一…

推荐两款github敏感信息搜集工具(gsil、gshark)

推荐两款github敏感信息搜集工具&#xff08;gsil、gshark&#xff09; - 云社区 - 腾讯云 (tencent.com) github敏感信息泄露是很多企业时常忽视的一个问题&#xff0c;国外有一份研究报告显示&#xff0c;在超过24,000份的GitHub公开数据中&#xff0c;发现有数千个文件中可能…

ChatGPT已打破图灵测试,新的测试方法在路上

生信麻瓜的 ChatGPT 4.0 初体验 偷个懒&#xff0c;用ChatGPT 帮我写段生物信息代码 代码看不懂&#xff1f;ChatGPT 帮你解释&#xff0c;详细到爆&#xff01; 如果 ChatGPT 给出的的代码不太完善&#xff0c;如何请他一步步改好&#xff1f; 全球最佳的人工智能系统可以通过…

在windows配置redis的一些错误及解决方案

目录 Unable to connect to Redis; nested exception is io.lettuce.core.RedisConnectionException:用客户端Redis Desktop Manager一样的密码端口&#xff0c;是可以正常连接的&#xff0c;但是运行java程序之后使用接口请求就会报错 Unable to connect to Redis; nested e…

使用正则表达式 移除 HTML 标签后得到字符串

需求分析 后台返回的数据是 这样式的 需要讲html 标签替换 high_light_text: "<span stylecolor:red>OPPO</span> <span stylecolor:red>OPPO</span> 白色 01"使用正则表达式 function stripHTMLTags(htmlString) {return htmlString.rep…