【数据结构与算法】【算法思想】回溯算法

贪心算法
回溯算法
分治算法
动态规划

回溯算法思想应用广泛,除了用来指导深度优先搜索这种经典算法设计之外,还可以用在如正则表达式匹配编译原理中的语法分析等。

除此之外,很多经典的数学问题都可以用回溯算法解决,比如数独,八皇后,0-1背包,图的着色,旅行商问题,全排列等等。

一:如何理解“回溯算法”

回溯的处理思想,有些类似枚举搜索。枚举所有的解,找到满足期望的解。为了有规律的枚举所有可能的解,避免遗漏和重复,所以把问题求解的过程分为多个阶段。每个阶段都有多个选择,先随意选择一个,当发现走不通时,就退回到上一个岔路口,另选一种走法继续走。
常用在“搜索”问题:在一组可能的解中,搜索满足期望的解

二:例子

八皇后问题

有一个8X8的棋盘,希望往里放8个棋子,每个棋子所在的行,列,对角线都不能有另一个棋子。

第一幅图满足要求,第二幅度图不满足,八皇后问题就是期望找到所有满足这种要求的的放棋子的方式。
我们把这个问题划分成8个阶段,依次将8个棋子放到第一行,第二行,第三行……第八行。


int[] result = new int[8];//全局或成员变量,下标表示行,值表示queen存储在哪一列
public void cal8queens(int row) { // 调用方式:cal8queens(0);if (row == 8) { // 8个棋子都放置好了,打印结果printQueens(result);return; // 8行棋子都放好了,已经没法再往下递归了,所以就return}for (int column = 0; column < 8; ++column) { // 每一行都有8中放法if (isOk(row, column)) { // 有些放法不满足要求result[row] = column; // 第row行的棋子放到了column列cal8queens(row+1); // 考察下一行}}
}private boolean isOk(int row, int column) {//判断row行column列放置是否合适int leftup = column - 1, rightup = column + 1;for (int i = row-1; i >= 0; --i) { // 逐行往上考察每一行if (result[i] == column) return false; // 第i行的column列有棋子吗?if (leftup >= 0) { // 考察左上对角线:第i行leftup列有棋子吗?if (result[i] == leftup) return false;}if (rightup < 8) { // 考察右上对角线:第i行rightup列有棋子吗?if (result[i] == rightup) return false;}--leftup; ++rightup;}return true;
}private void printQueens(int[] result) { // 打印出一个二维矩阵for (int row = 0; row < 8; ++row) {for (int column = 0; column < 8; ++column) {if (result[row] == column) System.out.print("Q ");else System.out.print("* ");}System.out.println();}System.out.println();
}

0-1背包

0-1背包是非常经典的算法问题,很多场景都可以抽象炒年糕这个问题模型。这个问题的经典解法是动态规划,不过还有一种简单但没有那么高效的解法,就是回溯算法。
0-1背包问题有很多变体,例如:有一个背包,背包总的承载重量是Wkg。现在有n个物品的重量不等,并且不可分割。我们现在期望选择几件物品,装载到背包中,在不超过所能装载重量的前提下,如何让背包中物品的总重量最大?
这个背包问题,在贪心算法中有提到,但在贪心算法中讲的物品是可以分割的,可以将物品的一部分装进背包。此处的背包问题,物品是不可分割的,要么装要么不装,所以叫0-1背包问题。
对于每个物品来说,都有两种选择,装进背包或在着不装进背包。对于n个物品来说,中的装法就有2^n种,去掉总重量超过过W kg的,从剩下的装法中选择中重量最接近W kg的。

可以用回溯的方法,不重复的找到2^n种装法。把物品依次排列,整个问题就分解为了n个阶段,每个阶段对应的一个物品怎么选择。先对第一个物品进行处理,选择装进去或是不装进去,然后在递归地处理剩下的物品。


public int maxW = Integer.MIN_VALUE; //存储背包中物品总重量的最大值
// cw表示当前已经装进去的物品的重量和;i表示考察到哪个物品了;
// w背包重量;items表示每个物品的重量;n表示物品个数
// 假设背包可承受重量100,物品个数10,物品重量存储在数组a中,那可以这样调用函数:
// f(0, 0, a, 10, 100)
public void f(int i, int cw, int[] items, int n, int w) {if (cw == w || i == n) { // cw==w表示装满了;i==n表示已经考察完所有的物品if (cw > maxW) maxW = cw;return;}f(i+1, cw, items, n, w);//不选择当前物品,直接考虑下一个(第 i+1 个),故 cw 不更新if (cw + items[i] <= w) {// 已经超过可以背包承受的重量的时候,就不要再装了f(i+1,cw + items[i], items, n, w);//表示选择了当前物品,故考虑下一个时,cw 通过入参更新为 cw + items[i]}
}

2 正则表达式

正则表达式里最重要的一种算法思想就是回溯。
正则表达式中,最重要的就是通配符,通配符结合在一起,可以表达非常丰富的语义。假设:正则表达式中只包含 “”和“?”这两种通配符,并且对这两种通配符的语义稍微做些改变。其中,“”匹配任意多个(大于等于0个)任意字符,“?”匹配零个或者一个任意字符。基于以上背景假设,看下如何用回溯算法,判断一个给定的文本,能否跟给定的正则表达式匹配?
依次考察正则表达式中的每个字符,当是非通配符时,就直接跟文本的字符进行匹配,如果相同,则继续往下处理;如果不同,则回溯。
如果遇到特殊字符的时候,就有多种处理方式,如“*”有多种匹配方案,可以匹配任意个文本串中的字符,我们就先随意的选择一种匹配方案,然后继续考察剩下的字符。如果中途发现无法继续匹配下去,就回到这个岔路口,重新选择一种匹配方案,然后继续匹配剩下的字符。


public class Pattern {private boolean matched = false;private char[] pattern; // 正则表达式private int plen; // 正则表达式长度public Pattern(char[] pattern, int plen) {this.pattern = pattern;this.plen = plen;}public boolean match(char[] text, int tlen) { // 文本串及长度matched = false;rmatch(0, 0, text, tlen);return matched;}private void rmatch(int ti, int pj, char[] text, int tlen) {if (matched) return; // 如果已经匹配了,就不要继续递归了if (pj == plen) { // 正则表达式到结尾了if (ti == tlen) matched = true; // 文本串也到结尾了return;}if (pattern[pj] == '*') { // *匹配任意个字符for (int k = 0; k <= tlen-ti; ++k) {rmatch(ti+k, pj+1, text, tlen);}} else if (pattern[pj] == '?') { // ?匹配0个或者1个字符rmatch(ti, pj+1, text, tlen);rmatch(ti+1, pj+1, text, tlen);} else if (ti < tlen && pattern[pj] == text[ti]) { // 纯字符匹配才行rmatch(ti+1, pj+1, text, tlen);}}
}

回溯算法本质上就是枚举,优点在于其类似于摸着石头过河的查找策略,且可以通过剪枝少走冤枉路。它可能适合应用于缺乏规律,或我们还不了解其规律的搜索场景中。

笔记整理来源: 王争 数据结构与算法之美

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

90 Subsets II

90 Subsets II leetcode第90题&#xff0c;用深度优先搜索的思路写出了三种不同的代码。一道题目&#xff0c;同样是深度优先搜索的问题&#xff0c;因为看问题角度不同&#xff0c;思路不同&#xff0c;代码也不一样 /*** [1,2,2,3] [] [1] [1,2] [1,2](重复) 情况&#xff1a…

2019 DDCTF 部分writeup

网上的wp已经很多了&#xff0c;但wp普遍很简略。我尽量写的详细一点。 一、WEB 滴~ 拿到题目后首先右键查看源代码&#xff0c;发现图片是以base64传送的 而且看url发现里面应该是包含了文件名&#xff0c;并且用了某个编码。测试过后是转16进制ascii码后两层bases64 &#xf…

第三十六期:如果把线程当作一个人来对待,所有问题都瞬间明白了

7月8日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 多线程的问题都曾经困扰过每个开发人员&#xff0c;今天将从全新视角来解说&#xff0c;希望读者都能明白。 强烈建议去运行下文章中的示例代码&#xff0c;自己体会下。 问题究竟出在哪里&#xff1f; 一个…

[Leetcode][第322题][JAVA][零钱兑换][回溯][记忆化搜索][动态规划]

【问题描述】[中等] 【解答思路】 1. 递归&#xff08;超时&#xff09; class Solution {int res Integer.MAX_VALUE;public int coinChange(int[] coins, int amount) {if(coins.length 0){return -1;}findWay(coins,amount,0);// 如果没有任何一种硬币组合能组成总金额&…

用BST解决729. My Calendar I 731. My Calendar II 732. My Calendar III

My Calendar的book方法实现指定开始时间、结束时间&#xff0c;在重叠次数要求不同的情况下怎么实现。 729 My Calendar I 要求任意两个事件之间不能重叠。如果要插入的事件和已经插入的事件不重叠&#xff0c;则插入&#xff1b;否则不插入。 731 MyCalendar II 要求任意三个…

[转载]抓大放小,要事为先

最近看到了swing框架&#xff0c;细节颇多。细细看来效率很低&#xff0c;偶然找到了这篇文章&#xff0c;恩确实有道理&#xff1b;以下&#xff1b; 对待人生的任何事情都要&#xff1a;抓大放小&#xff0c;要事为先 对于一个以前从来没有接触过java的人&#xff0c;java无疑…

第三十七期:如果你这样回答“什么是线程安全”,面试官都会对你刮目相看

6月12日 以下文章来源于编程新说 &#xff0c;作者编程新说李新杰 有读者跟我说&#xff0c;喜欢看我的文章&#xff0c;说很容易读&#xff0c;我确实在易读性上花费的心思不亚于在内容上。因为我不喜欢一上来就堆很多东西&#xff0c;而且把简单的东西搞得复杂人人都会&…

第三十八期:如何在Windows 10上使用Windows Update目录驱动程序安装打印机

尽管Win10能够自动设置大多数打印机&#xff0c;但有时Windows Update可能会在安装设备驱动程序时遇到问题-尤其是在添加较旧的打印机时。 作者&#xff1a;佚名来源&#xff1a;Win10系统之家 图片来源&#xff1a;伏天氏(m.futianshuwu.com) 伏天书屋(futianshuwu.com)10月…

MySQL 表和列的注释

像代码一样&#xff0c;可以为表以及表中的列添加注释&#xff0c;方便其他人知晓其功能。对于一些字段&#xff0c;在经过一定时间后&#xff0c;创建者未必也能想起其具体的含意&#xff0c;所以注释显得尤为重要。 注释的添加 注释的添加是通过在定义表或列的时候在末尾加上…

376 Wiggle Subsequence 贪心解法以及证明

376. Wiggle Subsequence 题目理解 给定一个数组&#xff0c;相邻两个数计算差值。差值排成的序列是正负相间的&#xff0c;那这个数组就是一个wiggle 数组。例如数组[1,7,4,9,2,5]&#xff0c;差值序列是(6,-3,5,-7,3)。原数组用坐标轴表示如下。 思路是&#xff1a;在一段…

【数据结构与算法】【算法思想】动态规划

贪心算法 回溯算法 分治算法 动态规划 贪心&#xff1a;一条路走到黑&#xff0c;就一次机会&#xff0c;只能哪边看着顺眼走哪边 回溯&#xff1a;一条路走到黑&#xff0c;无数次重来的机会&#xff0c;还怕我走不出来 (Snapshot View) 动态规划&#xff1a;拥有上帝视角&am…

第六十七期:Python爬虫44万条数据揭秘:如何成为网易音乐评论区的网红段子手

获取数据,其实逻辑并不复杂&#xff1a;爬取歌单列表里的所有歌单url、进入每篇歌单爬取所有歌曲url&#xff0c;去重、进入每首歌曲首页爬取热评&#xff0c;汇总。 作者&#xff1a;嗨学python来源&#xff1a;今日头条 获取数据 其实逻辑并不复杂&#xff1a; 爬取歌单列…

第一阶段SCRUM冲刺 08

昨天的成就&#xff1a;实现任务查找模块。感觉到了硬件支持对软件编程的重要性。 遇到的难题&#xff1a;电脑出现卡顿&#xff0c;编程工具十分卡&#xff0c;重启电脑好几次解决这个问题。电脑也是需要休息的。 今天的任务&#xff1a;进行资料上传模块&#xff1b;发布任务…

array专题7

714 Best Time to Buy and Sell Stock with Transaction Fee 思路 首先是暴力枚举。考虑在第idx天能做的操作&#xff1a;买&#xff1f;卖&#xff1f;不操作&#xff1f; /*** 暴力枚举* * param prices* param fee* return*/public int maxProfitV99(int[] prices, int f…

第三十九期:收藏 | 第一次有人把“分布式事务”讲的这么简单明了

不知道你是否遇到过这样的情况&#xff0c;去小卖铺买东西&#xff0c;付了钱&#xff0c;但是店主因为处理了一些其他事&#xff0c;居然忘记你付了钱&#xff0c;又叫你重新付。 作者&#xff1a;咖啡拿铁来源 又或者在网上购物明明已经扣款&#xff0c;但是却告诉我没有发…

【小技巧】字符char与整型int的相互转换

char转int char与int的相互转化&#xff0c;联想ASCII码&#xff0c;字符‘0’对应的值为48&#xff0c;所以不能直接加减‘ ’ char ch9; int ch_intch-0;//此时ch_int9int转char int i9&#xff1b; char i_chi0;//此时i_ch9必须牢记的ASCII

array专题8

670 Maximum Swap 思路&#xff1a;先把整数分解成一个一个的数&#xff0c;从0-n放着从最低位到最高位的数字。例如2376变成数组[6,7,3,2]。假设要替换的是最高位n-1,从0到n-2中查找是否有比nums[n-1]大的元素&#xff1b;如果有则替换&#xff0c;否则继续考虑替换n-2位。比…

nginx 编译安装

安装 Nginx 所依赖的基本服务&#xff1a; yum -y install gcc gcc-c automake pcre pcre-devel zlip zlib-devel openssl openssl-devel 官网 http://nginx.org 下载安装包 编译nginx make (可参考http://nginx.org/en/docs/configure.html) ./configure --prefix/data1/se…

第四十期:九个对Web开发者最有用的Python包,掌握这些,工资至少能涨涨

Matplotlib&#xff0c;正如其名称所暗示的那样&#xff0c;是一个用来绘制数学函数和模型的库;扩展了Numpy的作用&#xff0c;Matplotlib可以只用几行代码来创建图&#xff0c;条形图&#xff0c;散点图等诸多视觉表现。 作者&#xff1a;Python之眼来源&#xff1a;今日头条…

array专题9

新的一周&#xff0c;新的专题。array的中等难度的题目快要结束了。能感觉到进步&#xff0c;也依然能感觉到吃力。加油。 31 Next Permutation 思路&#xff1a;读懂了题意&#xff0c;知道是要求下一个排列数是什么。如果已经到最后一个了&#xff0c;那就返回最小的那个。…