首页 > 其他分享 >搜索优化

搜索优化

时间:2024-02-15 11:23:28浏览次数:29  
标签:终点 ed 短路 搜索 长度 优化 我们

主要有:剪枝、双向搜索、启发式搜索、迭代加深。

【剪枝】

拼接正方形

一道水题。基本想法:枚举每根木棍属于哪条边,\(O(\frac{4^m}{4!})\)。

但是这太慢了,我们剪枝:如果一条边目前的累计长度 \(>\frac{sum}{4}\),就返回 false。

于是我们就通过了。

运动员最佳匹配问题

先可以暴力枚举每个男匹配哪个女。

然后我们考虑优化:

如果对于两个匹配:\((i,k),(j,l)\),记其贡献为 \(S_1\);\((i,l),(j,k)\),记其贡献为 \(S_2\)。

若 \(S_1>S_2\),搜索时我们就不必搜索第二种情况了,因为我们交换一下可以更优。

若 \(S_1=S_2\),搜索时我们只需要搜一种即可。

code

【双端队列广搜】

Switch the Lamp On

想法:

对于每一个方格,我们把它的两组对角顶点连边。若一组顶点本来就被电线连接,其边权为 0;否则边权为 1。

发现此题边权只有两种,我们引入一种新的广搜:双端队列广搜。

考虑我们原本广搜求最短路的条件:无边权。(一种边权)

这是因为我们需要保证当第一次搜索到一个点的时候,当前距离一定刚好是最短路。

而现在我们可以用双端队列,保证我们一定是先搜完 0 的点、再搜完 1 的点。

具体而言,当我们遇到一个 0 的点,把它从队头入队;否则从队尾入队。

这样我们的搜索过程就像先把 0 的点给 floodfill 了一样。

0 的点在我们搜到的时候一定是最短路,而 1 的点一定在我们搜完所有 0 的点才会到,因此我们搜到的点第一次一定是最短的。

【双向广搜】

最接近目标值的子序列

如果我们暴力枚举每一个元素取或不取,\(2^{40}\) 不可接受。

但是,我们可以考虑把序列分成两半:先求出前 \(n/2\) 的所有子序列和,然后在搜索后一半的时候和前面搜出来的结果加起来,然后求得对应的答案。

具体而言,我们先 \(srh1()\) 求出前 \(n/2\) 个数的所有子序列和,存在一个 map 里。然后在我们 \(srh2()\) 每次搜到底的时候,我们通过 \(lower\_bound\) 找出 map 里面和本次搜索得出的 \(sum\) 加起来最接近 \(k\) 的元素,最后用这两个数的和与 \(k\) 的距离更新 \(ans\) 即可。

注(法二):

我们也可以把两次搜索的和们分别存进两个数组里面,然后用双指针扫一遍求最小距离。

code 法一

code 法二

迷宫最短路

这是一道经典而简单的问题,但是我们要用双向广搜来写。

我们可以考虑建立两个队列,一个代表从起点搜,一个代表从终点搜。另外我们也建立两个 \(dis\) 数组,一个是从起点出发的距离,一个是从终点出发的距离。

当我们搜索的时候,每次都同时搜出起点和终点的下一层。并且当我们在搜索起点的时候,如果遇到一个终点搜过的格子,我们就可以把这个格子到起点、终点的最短路长度加起来求出从起点到终点的最短路长度。

正确性是显然的。因为无论是起点还是终点,我们都在做正常的广搜,所以每一个格子被搜的时候其最短路长度已经确定,并且我们的层数是从小到大搜的。
因此,当我们起点碰到终点的时候,我们可以确定这个格子以后不会出现更短的路径。

注意处理 起点=终点 的特殊情况。

code(双向)


双向广搜为什么能加快速度?

我们做正常广搜的时候,相当于在起点画圆,不断扩大半径,直到终点被包括进来。
双向广搜时,我们相当于同时在起点和终点画圆,同时扩大半径,直到两个圆有交点。

因为我们从起点和终点一起搜,所以圆的半径减半,那么 面积就会变成 \(\frac{1}{4}\)。
虽然题目中可能并没有画圆一样整齐的扩张,但是双向搜索仍能减少复杂度。

【启发式搜索】

启发式:任何不影响正确性的优化,都可以叫做启发式。

【启发式剪枝】

小木棍

最初想法:

枚举原始木棍长度,全排列枚举,参数里面记录当前木棍的剩余长度。

优化:

  1. 原始木棍长一定是总长的因数;

  2. 避免同组内的木棍又排列一遍重复,令选择时必须比组内上一根木棍短;

  3. 组间重复,令每个组的第一根木棍长度递减;

  4. 重复数,如果选了当前木棍不行,就不要选和当前木棍同长度的木棍了,这可以用 cnt 数组完成,每次搜索循环所有长度,如果还有该长度木棍就尝试,不行直接循环下一种长度;

  5. 如果当前剩余的长度可以刚好和一根木棍匹配,一定要用这个木棍,如果用了这个木棍后返回不能拼成,一定是之前的木棍选错了,这是因为如果不选这个木棍,一定需要用几根更小的拼出来,不如直接用这个木棍;

  6. 如果当前木棍当第一根时不行,一定是之前选错了。

加上这些优化,就比较快了。

【A*】

对于一个人,我们在枚举答案的时候一定会想要从那些 “看起来更可能是答案” 的方向枚举。
但是计算机并不理解我们的想法,于是我们现在要尝试让计算机也理解我们的想法。

以找棋盘最短路为例。

我们可以建立一个估值函数 \(f(x)\),表示从 \(st\) 经过位置 \(x\) 走到 \(ed\) 的估计最短路长度。当这个 \(f(x)\) 越小,就代表我们越可能通过位置 \(x\) 走到最短路。然后我们在广搜的时候使用优先队列,每次取出 \(f(x)\) 最小的更新。

我们令 \(f(x)=g(x)+h(x)\),其中 \(g(x)\) 是我们已经搜出来的从 \(st\) 到 \(x\) 的长度,\(h(x)\) 是我们估计从 \(x\) 走到 \(ed\) 的最短路长度。

比如,如果我们把 \(h(x)\) 定义为 “\(x\) 到 \(ed\) 的曼哈顿距离”,那么我们就会优先朝 \(ed\) 所在的方向搜索。


上面我们说明了启发式搜索可以让我们的搜索更快,下面我们看看如何保证正确性。

正确性: \(h(x)\leq\) 实际 \(x\rightarrow ed\) 的距离 \(w(x)\)。

实际上,启发式已经破坏了广搜的正确性。

因为我们广搜的正确性是因为:当我们搜索 \(k\) 长度的时候,已经搜完了 \(1\sim k-1\) 的路径,所以如果在 \(k\) 长度第一次碰到终点,\(k\) 一定是最短长度。

但是,我们启发式可能朝靠近 \(ed\) 的方向搜到了长度 \(10\) 时碰到了终点,但是在反向可能只搜到了长度 \(5\),我们如何保证另一个方向不会有其他方法在长度小于 \(10\) 时碰到终点?

假如我们能保证 \(h(x)\leq w(x)\),那么我们就有 \(g(x)+h(x)\leq g(x)+w(x)\),即 \(f(x)\leq g(x)+w(x)\)(经过 x 走到终点的最短路估值 \(\leq\) 实际上经过 x 走到终点的最短路长度)

而我们会优先朝靠近 \(ed\) 的方向搜,肯定是因为那边的 \(f()\) 更小,所以 \(f(ed)\leq f(x)\),即:
从靠近 \(ed\) 的方向走到 \(ed\) 的最短路长度估值 \(\leq\) 从不靠近 \(ed\) 的方向走到 \(ed\) 的最短路长度估值。

而上面又有 \(f(x)\leq w(st)\),所以 \(f(ed)\leq g(x)+w(x)\),即从靠近 \(ed\) 的方向去 \(ed\) 比从不靠近 \(ed\) 的方向走 更短。

注意 \(f(ed)=g(ed)+h(ed)\),因为 \(ed\) 就是终点,所以 \(h(ed)=0\),所以 \(f(ed)=g(ed)\) 就是 从靠近 \(ed\) 的方向去 \(ed\) 的最短距离。

而这个最短距离已经被证实比其他不走这个方向的最短距离 更短,所以这个距离就是真正的最短距离。

综上所述,只要满足 \(h(x)\leq\) 实际 \(x\rightarrow ed\) 的距离,我们第一次搜到终点得到的距离一定就是最短距离。


还要注意一种特殊情况:

? 5 ...
3 4 5
... ... ...

如图,我们现在搜到了 "3" 的格子,那么 “?” 的最短距离应当是 4,但是有可能因为 ? 的 \(h()\) 比其他方向更差,所以在加入优先队列后被其他几个邻格给挤到后面去了,没有第一时间搜到。而我们已经走其他方向把 ? 的距离更新为 6 了。而此时 ? 的最短距离就错误了。

但是我们不需要担心我们会用这个错误的距离更新终点导致答案错误。

  1. 如果 ? 就是终点,则 \(h(?)=0\),而我们上面没有第一时间取出 ?的原因是因为 \(h(?)\) 更差,矛盾;

  2. 如果 ? 用来更新终点,那么对于我们在 3 更新时加入的 ? 状态,它的 \(f(?)=g(?)+h(?)=4+h(?)\),而如果我们用了 \(f(?)=6+h(?)\) 的状态去更新终点,它比 \(4+h(?)\) 还要大,那么我们肯定在这之前就已经用了 \(4+h(?)\) 的状态更新过一遍了。

我们尝试推广第二种情况:

假如我们有两条去终点的路径,但是用了更长的那一条去更新了终点。

在更短的那一条路上取一个没有在另一条路径上的点 \(x\)。显然更长的那条路径长度 \(a\geq\) 实际最短路长度。

而 \(f(x)=g(x)+h(x)\leq\) 实际上的最短路长度。(因为 \(h(x)\leq\) 实际)

所以 \(f(x)\leq\) 最短路长度 \(\leq a\),那我们应该在更早的时候就已经用 \(x\) 更新了终点了。

迷宫最短路

这题也可以用 A* 做。我们定义 \(h(x)\) 为 \(x\) 和 \(ed\) 的曼哈顿距离,满足 \(h(x)\leq\) 实际 \(x\rightarrow ed\) 的距离。

code(A*)

【迭代加深】

众所周知,深搜的特点是一直往底走,走到底了再回头。

但是,如果我们遇到了一道没有底的题,也就是结束条件模糊的题,我们应该怎么用深搜做?

这个时候,我们就可以用迭代加深搜索。

我们可以人为限定一个最大深度,如果深搜的深度大于它,我们就认为在这个深度下,我们这么走是没有答案的,然后回溯。

每次将限定的深度增大一点,直到在限定深度内搜到答案。因为搜索树大小是指数级增长,所以之前搜索的时间可以当做一个较大的常数来看。

埃及分数

这题的特点在于我们不知道一共需要几个分数加起来,导致我们可能会搞出一大堆分母特别大的分数,然后无限往下搜。
于是我们循环需要的分数个数搜索。

code

标签:终点,ed,短路,搜索,长度,优化,我们
From: https://www.cnblogs.com/FLY-lai/p/18016076

相关文章

  • 斜率优化
    以P5785[SDOI2012]任务安排为例朴素方程其实也没那么简单,第一眼想法是设\(f(i,k)\)表示以\(i\)为结尾,共分了\(k\)段的总方案数\[f(i,k)=\min_{j=0}^{i-1}f(j,k-1)+(sumc_i-sumc_j)\cdot(sumt_i+s\cdotk)\]枚举的\(j\)表示当前选的第\(k\)段为\((j,i]\)\(O(n......
  • 电子商务发货流程如何优化?发货单和快递单如何匹配?
     目前采用的是快递单打印订单备注的问题解决打包问题,但是填写备注的工作量巨大还容易出错,现在成熟的解决方案是什么样的? 电子商务的发货流程如下:库存组依据账物组交给的销售定单进行配货,配货结束在配货单上签字确认后交给发货组。发货组接到配货组交给的物品后依据销售定单......
  • GDI+性能优化
    每个Windows控件都可以拥有一个paint事件处理程序和一个表示此控件是绘图画布的Graphics对象。这意味着我们可以使用一个按钮或一个列表框作为绘图画布。如果在菜单或按钮的Click事件处理程序中绘制图形对象,则必须最后调用 this.Invalidate()方法。如果不调用,窗体将不......
  • 力扣递归 深度优先搜索 之 104. 二叉树的最大深度
    给定一个二叉树root,返回其最大深度。二叉树的最大深度是指从根节点到最远叶子节点的最长路径上的节点数。 示例1: 输入:root=[3,9,20,null,null,15,7]输出:3示例2:输入:root=[1,null,2]输出:2/** *Definitionforabinarytreenode. *publicclassTre......
  • Python 机器学习 线性回归 梯度下降法优化损失函数
    ​ Python机器学习中,梯度下降法是一种用于优化线性回归模型(以及其他机器学习算法)的损失函数的通用算法。目的是通过迭代地调整模型的参数(权重和截距),以最小化损失函数,例如均方误差(MSE)。梯度下降的基本思想是计算损失函数相对于每个参数的梯度(即偏导数),然后朝着减少损失的方向调......
  • 力扣回溯 深度优先搜索 dfs 之 17. 电话号码的字母组合
    给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按任意顺序返回。给出数字到字母的映射如下(与电话按键相同)。注意1不对应任何字母。 示例1:输入:digits="23"输出:["ad","ae","af","bd","be","bf","cd","ce","cf&qu......
  • 力扣回溯 深度优先搜索dfs之78. 子集
    给你一个整数数组 nums,数组中的元素互不相同。返回该数组所有可能的子集(幂集)。解集不能包含重复的子集。你可以按任意顺序返回解集。 示例1:输入:nums=[1,2,3]输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]示例2:输入:nums=[0]输出:[[],[0]] classSol......
  • NET项目&DLL反编译&MSSQL监控&VS搜索&注入&上传
    知识点1.NET普通源码&编译源码2.DLL反编译&后缀文件&指向3.代码审计-SQL注入&文件上传ASPX文件->CSASPX.CSDLL反编译后寻找看核心代码分析漏洞CSASPX.CSDLL反编译文件->ASPX文件寻找确定漏洞进行调试测试代码审计时要把这个反编译文件提取导入到IDE中后期搜索关......
  • 搜索引擎用法 cheatsheet
    逻辑写法与keyword1keyword2或keyword1ORkeyword2限定关键词的排列"keyword"限定搜索的网站site:cnblogs.comsite:cnblogs.com/Undefined443site:.com只搜索标题intitle:keywordallintitle:keyword1keyword2只搜索网页正文intext:keywordallint......
  • 搜索
    一.BFS&DFS基础搜索是暴力法的具体体现,具有简单直接的特点,利用了计算机强大的计算能力。(也是用来混部分分的绝佳利器)搜索的基础算法分为两处:宽度优先搜索(又叫广度优先搜索,即BFS),深度优先搜索(DFS)思维区别BFS:“全面扩散,逐层递进”DFS:“一路到底,逐步回退”DFS代码框架voiddfs......