【软件设计师】常见的算法设计方法——穷举搜索法

简介: 【软件设计师】常见的算法设计方法——穷举搜索法

🐓 穷举搜索法

什么是穷举搜索法

穷举搜索法,又称枚举法穷举法,是一种编程中常用到的问题求解方法。当找不到解决问题的规律时,穷举搜索法会对可能是解的众多候选解按某种顺序进行逐一枚举和检验,并从中找出那些符合要求的候选解作为问题的解。

穷举搜索法的基本思想是列举出所有可能的情况,逐个判断哪些情况符合问题所要求的条件,从而得到问题的全部解答。这种方法利用计算机运算速度快、精确度高的特点,对要解决问题的所有可能情况,一个不漏地进行检查,从中找出符合要求的答案。


穷举搜索法的基本场景

穷举搜索法的基本场景通常涉及以下两个方面:

问题所涉及的情况:在穷举搜索法中,首先需要明确问题所涉及的所有可能情况。这些情况的种数必须可以确定,并且需要被一一列举出来。既不能重复,也不能遗漏。例如,如果问题是求解一个特定数学方程的所有整数解,那么就需要列举出所有可能的整数,并检查哪些整数满足方程的条件。


答案需要满足的条件:在列举出所有可能的情况后,需要分析这些情况需要满足什么条件才能成为问题的答案。这些条件可能包括数学公式、逻辑关系、约束条件等。例如,在求解数学方程的情况下,答案需要满足方程等于零的条件。


穷举搜索法的基本原理

穷举搜索法的基本原理是通过列举所有可能的情况来找到满足特定条件的解。这种方法基于这样一个概念:如果一个问题的解空间是有限的,那么通过逐一检查每一个可能的解,最终一定能够找到所有满足条件的解。


算法设计——穷举搜索法


🐓 代码实例解析

设计思路

使用穷举法解决问题,基本上就是以下两个步骤:

确定问题的解(或状态)的定义、解空间的范围以及正确解的判定条件;

根据解空间的特点来选择搜索策略,逐个检验解空间中的候选解是否正确;


解空间(范围内找解)

解空间就是全部可能的候选解的一个约束范围,确定问题的解就在这个约束范围内,将搜索策略应用到这个约束范围就可以找到问题的解。


剪枝策略

对解空间穷举搜索时,如果有一些状态节点可以根据问题提供的信息明确地被判定为不可能演化出最优解,也就是说,从此节点开始遍历得到的子树,可能存在正确的解,但是肯定不是最优解,就可以跳过此状态节点的遍历,这将极大地提高算法的执行效率,这就是剪枝策略,应用剪枝策略的难点在于如何找到一个评价方法(估值函数)对状态节点进行评估。


案例1

鸡兔同笼问题。有鸡和兔在一个笼子中,数头共 50 个头,数脚共 120 只脚,问:鸡和兔分别有多少只?


情况一:盲目搜索

假设买 x 鸡,y 只兔,使用代码求解如下:

for x in range(1, 51):
    for y in range(1, 51):
        if x + y == 50 and 2*x + 4*y == 120:
            print x, y
 


情况二:启发搜索

假设买 x 鸡,y 只兔,使用代码求解如下:

for x in range(1, 51):
    if 2*x + 4*(50-x) == 120:
        print x, 50-x


案例2

找出给定整数数组中所有可能的两个数之和的组合

import java.util.ArrayList;  
import java.util.List;  
  
public class ExhaustiveSearchExample {  
  
    public static void main(String[] args) {  
        int[] numbers = {1, 2, 3, 4, 5};  
        List<int[]> allSumCombinations = findAllSumCombinations(numbers, 10);  
        for (int[] combination : allSumCombinations) {  
            System.out.println("(" + combination[0] + ", " + combination[1] + ")");  
        }  
    }  
  
    public static List<int[]> findAllSumCombinations(int[] numbers, int targetSum) {  
        List<int[]> combinations = new ArrayList<>();  
        for (int i = 0; i < numbers.length; i++) {  
            for (int j = i + 1; j < numbers.length; j++) {  
                if (numbers[i] + numbers[j] == targetSum) {  
                    combinations.add(new int[]{numbers[i], numbers[j]});  
                }  
            }  
        }  
        return combinations;  
    }  
}


代码运行结果及解释

(1, 9)  
(2, 8)  
(3, 7)  
(4, 6)  
(5, 5)


findAllSumCombinations方法使用了两层循环来穷举数组中所有可能的数对。对于每一对数(numbers[i]numbers[j]),它检查它们的和是否等于targetSum。如果等于,它就将这对数添加到一个列表中。最后,该方法返回这个列表,其中包含了所有满足条件的数对。


🐓 穷举搜索法的优缺点及注意事项

穷举搜索法的优点

直观易懂:穷举搜索法通常基于问题的直接描述,易于理解和实现。

正确性保证:由于穷举搜索法会检查所有可能的解,因此它能够确保找到问题的最优解(如果存在的话)。

适用性广泛:这种方法适用于许多问题,特别是那些没有更简单或更有效算法可用的问题。


穷举搜索法的缺点

计算量大:穷举搜索法需要检查所有可能的解,因此计算量通常很大,尤其是在问题规模较大时。

效率低下:由于需要检查大量解,穷举搜索法通常非常耗时,可能在现实应用中不可行。

资源消耗多:在处理大规模问题时,穷举搜索法可能需要大量的内存和计算资源。


使用穷举搜索法需要注意的问题

问题规模:在使用穷举搜索法之前,需要仔细评估问题的规模。如果问题规模太大,穷举搜索法可能不是一个好选择。

优化技巧:尽管穷举搜索法本身可能效率低下,但可以通过一些优化技巧(如剪枝、排序、哈希等)来减少计算量和提高效率。

利用问题特性:在某些情况下,可以利用问题的特性(如对称性、单调性等)来减少搜索空间或优化搜索过程。

考虑其他算法:如果穷举搜索法不切实际,可能需要考虑使用其他更高效的算法来解决问题

相关文章
|
2天前
|
存储 算法 Java
Java中,树与图的算法涉及二叉树的前序、中序、后序遍历以及DFS和BFS搜索。
【6月更文挑战第21天】Java中,树与图的算法涉及二叉树的前序、中序、后序遍历以及DFS和BFS搜索。二叉树遍历通过访问根、左、右子节点实现。DFS采用递归遍历图的节点,而BFS利用队列按层次访问。以下是简化的代码片段:[Java代码略]
10 4
|
2天前
|
存储 算法 Java
Java查找算法概览:二分查找适用于有序数组,通过比较中间元素缩小搜索范围;哈希查找利用哈希函数快速定位,示例中使用HashMap存储键值对,支持多值关联。
【6月更文挑战第21天】Java查找算法概览:二分查找适用于有序数组,通过比较中间元素缩小搜索范围;哈希查找利用哈希函数快速定位,示例中使用HashMap存储键值对,支持多值关联。简单哈希表实现未涵盖冲突解决和删除操作。
10 1
|
7天前
|
机器学习/深度学习 算法 C语言
详细介绍递归算法在 C 语言中的应用,包括递归的基本概念、特点、实现方法以及实际应用案例
【6月更文挑战第15天】递归算法在C语言中是强大力量的体现,通过函数调用自身解决复杂问题。递归涉及基本概念如自调用、终止条件及栈空间管理。在C中实现递归需定义递归函数,分解问题并设定停止条件。阶乘和斐波那契数列是经典应用示例,展示了递归的优雅与效率。然而,递归可能导致栈溢出,需注意优化。学习递归深化了对“分而治之”策略的理解。**
22 7
|
5天前
|
存储 算法 Java
广度优先搜索(Breadth-First Search,BFS)是一种用于图的遍历或搜索的算法。
广度优先搜索(Breadth-First Search,BFS)是一种用于图的遍历或搜索的算法。
|
8天前
|
算法 JavaScript 决策智能
基于禁忌搜索算法的TSP路径规划matlab仿真
**摘要:** 使用禁忌搜索算法解决旅行商问题(TSP),在MATLAB2022a中实现路径规划,显示优化曲线与路线图。TSP寻找最短城市访问路径,算法通过避免局部最优,利用禁忌列表不断调整顺序。关键步骤包括初始路径选择、邻域搜索、解评估、选择及禁忌列表更新。过程示意图展示搜索效果。
|
24天前
|
算法 数据安全/隐私保护 C++
基于二维CS-SCHT变换和扩频方法的彩色图像水印嵌入和提取算法matlab仿真
该内容是关于一个图像水印算法的描述。在MATLAB2022a中运行,算法包括水印的嵌入和提取。首先,RGB图像转换为YUV格式,然后水印通过特定规则嵌入到Y分量中,并经过Arnold置乱增强安全性。水印提取时,经过逆过程恢复,使用了二维CS-SCHT变换和噪声对比度(NC)计算来评估水印的鲁棒性。代码中展示了从RGB到YUV的转换、水印嵌入、JPEG压缩攻击模拟以及水印提取的步骤。
|
1天前
|
算法 Python
二维矩形件排样算法之最低水平线搜索算法实现
二维矩形件排样算法之最低水平线搜索算法实现
5 0
|
29天前
|
索引
浅谈两个重要的搜索算法
【5月更文挑战第15天】线性搜索从数组一端按顺序遍历,直到找到目标元素,平均和最坏情况的时间复杂度均为O(N)。二分查找适用于排序数组,通过比较中间元素快速定位目标,最佳、平均和最坏情况的时间复杂度都是O(logN)。
19 6
|
5天前
|
人工智能 算法 Java
深度优先搜索(Depth-First Search,DFS)是一种用于遍历或搜索树或图的算法。
深度优先搜索(Depth-First Search,DFS)是一种用于遍历或搜索树或图的算法。
|
9天前
|
算法
【经典LeetCode算法题目专栏分类】【第6期】二分查找系列:x的平方根、有效完全平方数、搜索二位矩阵、寻找旋转排序数组最小值
【经典LeetCode算法题目专栏分类】【第6期】二分查找系列:x的平方根、有效完全平方数、搜索二位矩阵、寻找旋转排序数组最小值