当前位置: 首页 > news >正文

网站建设对于电子商务的意义绵阳住房和城乡建设厅官方网站

网站建设对于电子商务的意义,绵阳住房和城乡建设厅官方网站,nas wordpress外网,静态网站建设课程设计算法训练DAY24|回溯1 第77题. 组合 力扣题目链接 给定两个整数 n 和 k#xff0c;返回 1 ... n 中所有可能的 k 个数的组合。 示例: 输入: n 4, k 2 输出: [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], ] 上面我们说了要解决 n为100#xff0c;k为50的情况#xff0…算法训练DAY24|回溯1 第77题. 组合 力扣题目链接 给定两个整数 n 和 k返回 1 ... n 中所有可能的 k 个数的组合。 示例: 输入: n 4, k 2 输出: [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], ] 上面我们说了要解决 n为100k为50的情况暴力写法需要嵌套50层for循环那么回溯法就用递归来解决嵌套层数的问题。 递归来做层叠嵌套可以理解是开k层for循环每一次的递归中嵌套一个for循环那么递归就可以用于解决多层嵌套循环的问题了。 此时递归的层数大家应该知道了例如n为100k为50的情况下就是递归50层。 一些同学本来对递归就懵回溯法中递归还要嵌套for循环可能就直接晕倒了 如果脑洞模拟回溯搜索的过程绝对可以让人窒息所以需要抽象图形结构来进一步理解。 我们在关于回溯算法你该了解这些 中说到回溯法解决的问题都可以抽象为树形结构N叉树用树形结构来理解回溯就容易多了。 那么我把组合问题抽象为如下树形结构 可以看出这棵树一开始集合是 1234 从左向右取数取过的数不再重复取。 第一次取1集合变为234 因为k为2我们只需要再取一个数就可以了分别取234得到集合[1,2] [1,3] [1,4]以此类推。 每次从集合中选取元素可选择的范围随着选择的进行而收缩调整可选择的范围。 图中可以发现n相当于树的宽度k相当于树的深度。 那么如何在这个树上遍历然后收集到我们要的结果集呢 图中每次搜索到了叶子节点我们就找到了一个结果。 相当于只需要把达到叶子节点的结果收集起来就可以求得 n个数中k个数的组合集合。 在关于回溯算法你该了解这些 中我们提到了回溯法三部曲那么我们按照回溯法三部曲开始正式讲解代码了。 #回溯法三部曲 递归函数的返回值以及参数 在这里要定义两个全局变量一个用来存放符合条件单一结果一个用来存放符合条件结果的集合。 代码如下 vectorvectorint result; // 存放符合条件结果的集合 vectorint path; // 用来存放符合条件结果 其实不定义这两个全局变量也是可以的把这两个变量放进递归函数的参数里但函数里参数太多影响可读性所以我定义全局变量了。 函数里一定有两个参数既然是集合n里面取k个数那么n和k是两个int型的参数。 然后还需要一个参数为int型变量startIndex这个参数用来记录本层递归的中集合从哪里开始遍历集合就是[1,...,n] 。 为什么要有这个startIndex呢 建议在77.组合视频讲解 (opens new window)中07:36的时候开始听startIndex 就是防止出现重复的组合。 从下图中红线部分可以看出在集合[1,2,3,4]取1之后下一层递归就要在[2,3,4]中取数了那么下一层递归如何知道从[2,3,4]中取数呢靠的就是startIndex。 所以需要startIndex来记录下一层递归搜索的起始位置。 那么整体代码如下 vectorvectorint result; // 存放符合条件结果的集合 vectorint path; // 用来存放符合条件单一结果 void backtracking(int n, int k, int startIndex) 回溯函数终止条件 什么时候到达所谓的叶子节点了呢 path这个数组的大小如果达到k说明我们找到了一个子集大小为k的组合了在图中path存的就是根节点到叶子节点的路径。 如图红色部分 此时用result二维数组把path保存起来并终止本层递归。 所以终止条件代码如下 if (path.size() k) {result.push_back(path);return; } 单层搜索的过程 回溯法的搜索过程就是一个树型结构的遍历过程在如下图中可以看出for循环用来横向遍历递归的过程是纵向遍历。 如此我们才遍历完图中的这棵树。 for循环每次从startIndex开始遍历然后用path保存取到的节点i。 代码如下 for (int i startIndex; i n; i) { // 控制树的横向遍历path.push_back(i); // 处理节点backtracking(n, k, i 1); // 递归控制树的纵向遍历注意下一层搜索要从i1开始path.pop_back(); // 回溯撤销处理的节点 } 可以看出backtracking递归函数通过不断调用自己一直往深处遍历总会遇到叶子节点遇到了叶子节点就要返回。 backtracking的下面部分就是回溯的操作了撤销本次处理的结果。 关键地方都讲完了组合问题C完整代码如下 class Solution { private:vectorvectorint result; // 存放符合条件结果的集合vectorint path; // 用来存放符合条件结果void backtracking(int n, int k, int startIndex) {if (path.size() k) {result.push_back(path);return;}for (int i startIndex; i n; i) {path.push_back(i); // 处理节点backtracking(n, k, i 1); // 递归path.pop_back(); // 回溯撤销处理的节点}} public:vectorvectorint combine(int n, int k) {result.clear(); // 可以不写path.clear();   // 可以不写backtracking(n, k, 1);return result;} }; 时间复杂度: O(n * 2^n) 空间复杂度: O(n) 还记得我们在关于回溯算法你该了解这些 中给出的回溯法模板么 如下 void backtracking(参数) {if (终止条件) {存放结果;return;} ​for (选择本层集合中元素树中节点孩子的数量就是集合的大小) {处理节点;backtracking(路径选择列表); // 递归回溯撤销处理结果} } 对比一下本题的代码是不是发现有点像 所以有了这个模板就有解题的大体方向不至于毫无头绪。 #总结 组合问题是回溯法解决的经典问题我们开始的时候给大家列举一个很形象的例子就是n为100k为50的话直接想法就需要50层for循环。 从而引出了回溯法就是解决这种k层for循环嵌套的问题。 然后进一步把回溯法的搜索过程抽象为树形结构可以直观的看出搜索的过程。 接着用回溯法三部曲逐步分析了函数参数、终止条件和单层搜索的过程。 #剪枝优化 我们说过回溯法虽然是暴力搜索但也有时候可以有点剪枝优化一下的。 在遍历的过程中有如下代码 for (int i startIndex; i n; i) {path.push_back(i);backtracking(n, k, i 1);path.pop_back(); } 这个遍历的范围是可以剪枝优化的怎么优化呢 来举一个例子n 4k 4的话那么第一层for循环的时候从元素2开始的遍历都没有意义了。 在第二层for循环从元素3开始的遍历都没有意义了。 这么说有点抽象如图所示 图中每一个节点图中为矩形就代表本层的一个for循环那么每一层的for循环从第二个数开始遍历的话都没有意义都是无效遍历。 所以可以剪枝的地方就在递归中每一层的for循环所选择的起始位置。 如果for循环选择的起始位置之后的元素个数 已经不足 我们需要的元素个数了那么就没有必要搜索了。 注意代码中i就是for循环里选择的起始位置。 for (int i startIndex; i n; i) { 接下来看一下优化过程如下 已经选择的元素个数path.size(); 还需要的元素个数为: k - path.size(); 在集合n中至多要从该起始位置 : n - (k - path.size()) 1开始遍历 为什么有个1呢因为包括起始位置我们要是一个左闭的集合。 举个例子n 4k 3 目前已经选取的元素为0path.size为0n - (k - 0) 1 即 4 - ( 3 - 0) 1 2。 从2开始搜索都是合理的可以是组合[2, 3, 4]。 这里大家想不懂的话建议也举一个例子就知道是不是要1了。 所以优化之后的for循环是 for (int i startIndex; i n - (k - path.size()) 1; i) // i为本次搜索的起始位置 优化后整体代码如下 class Solution { private:vectorvectorint result;vectorint path;void backtracking(int n, int k, int startIndex) {if (path.size() k) {result.push_back(path);return;}for (int i startIndex; i n - (k - path.size()) 1; i) { // 优化的地方path.push_back(i); // 处理节点backtracking(n, k, i 1);path.pop_back(); // 回溯撤销处理的节点}} public:vectorvectorint combine(int n, int k) {backtracking(n, k, 1);return result;} }; #剪枝总结 本篇我们准对求组合问题的回溯法代码做了剪枝优化这个优化如果不画图的话其实不好理解也不好讲清楚。 所以我依然是把整个回溯过程抽象为一棵树形结构然后可以直观的看出剪枝究竟是剪的哪里。 ##
http://www.pierceye.com/news/611056/

相关文章:

  • 适合这手机浏览器主页的网站wordpress本地上传服务器
  • 济南百度网站开发寮步镇做网站
  • 营销类型的公司网站专注高密做网站哪家好
  • 公司网站建设找谁做网络渠道
  • 网站建设公司 校园网站html5商城网站
  • 自学it做网站厦门网站推广¥做下拉去118cr
  • 汕头市做网站优化国内时事新闻
  • 网站文章来源seowordpress 搜索 分词
  • 网站建设和网络推广微信开发品牌
  • 湛江网站关键词优化百度推广优化技巧
  • 做盗版网站会怎样网页设计规范2018
  • 做个中英文网站多少钱网页设计图片作品
  • iis7 添加php网站网站为什么需要空间
  • 网站到首页排名h5怎么制作的
  • 网站制作教程 pdf下载培训网站制作网站
  • 网站开发文档范例国外服务器租用价格表
  • 六安网站制作费用怎么做百度提交入口网站
  • centos7做网站做pc端网站讯息
  • 驻马店建设网站安徽全过程网站搭建案例
  • 企业网站推广费用wordpress相册汉化版
  • 怎么做正规网站广告网站设计怎么样
  • 深圳营销型网站公司电话云渲染网站开发
  • 生成网站有吗免费的网站建设服务有哪些内容
  • 网站建设制作公司思企互联超级采购小程序怎么注册
  • 燕郊做网站找谁wordpress登录修改
  • 大概开发一个网站多少钱php做网站商城系统怎么样
  • wordpress网站程序员登录百度账号
  • wordpress trac网站优化公司哪家好
  • 网站建设cms系统抖音seo推广外包公司好做吗
  • 南宁商城网站建设logo设计网站生成器