数据结构与算法:常用的启发式算法

在数据结构的领域中,启发式算法是一类用于解决优化问题的算法,它们在每一步选择中都做出当前看来最好的选择,但并不保证总能找到全局最优解。这类算法广泛应用于资源分配、路径规划、存储分配等问题。以下是一些常用的启发式算法及其区别:

1.贪心算法(Greedy Algorithm)

贪心算法是一种在每一步选择中都采取当前最好或最优的选择,以期望结果是全局最好或最优的算法。贪心算法通常用于解决具有最优子结构特点的问题,例如最小生成树、哈夫曼编码等。

贪心算法的主要优点是实现简单、计算效率高。但缺点是可能会陷入局部最优,导致全局最优解不被找到。因此,在使用贪心算法时,需要先分析问题是否具有最优子结构特点,以及局部最优解是否一定能得到全局最优解。

示例:最小生成树

using System;
using System.Collections.Generic;public class GreedyAlgorithmExample
{public static List<int>[] FindMinimumSpanningTree(int n, int[,] weights){var mst = new List<int>[n];for (int i = 0; i < n; i++){mst[i] = new List<int>();}var visited = new bool[n];visited[0] = true;for (int i = 1; i < n; i++){int minWeight = int.MaxValue;int minEdge = -1;for (int j = 0; j < n; j++){if (!visited[j]){for (int k = 0; k < n; k++){if (!visited[k] && weights[j, k] < minWeight){minWeight = weights[j, k];minEdge = k;}}}}mst[i].Add(minEdge);mst[minEdge].Add(i);visited[minEdge] = true;}return mst;}public static void Main(){int n = 5;int[,] weights = {{0, 4, 0, 0, 1, 3},{4, 0, 8, 0, 0, 0},{0, 8, 0, 7, 0, 4},{0, 0, 7, 0, 9, 1},{1, 0, 0, 9, 0, 2},{3, 0, 4, 1, 2, 0}};var mst = FindMinimumSpanningTree(n, weights);// 输出最小生成树for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){if (mst[i].Contains(j)){Console.Write("{0} ", j);}}Console.WriteLine();}}
}

2. 动态规划(Dynamic Programming)

动态规划是一种通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法。动态规划的核心思想是保存子问题的解,以免重复计算。动态规划通常用于解决具有重叠子问题和最优子结构特点的问题,例如背包问题、最长公共子序列等。

动态规划的主要优点是能够保证找到全局最优解。但缺点是计算量较大,当问题规模较大时,时间复杂度和空间复杂度较高。因此,在使用动态规划时,需要正确划分子问题,并找到递推关系和边界条件。

示例:最长公共子序列(Longest Common Subsequence, LCS)

using System;public class DynamicProgrammingExample
{public static int FindLongestCommonSubsequence(string text1, string text2){int[,] dp = new int[text1.Length + 1, text2.Length + 1];for (int i = 1; i <= text1.Length; i++){for (int j = 1; j <= text2.Length; j++){if (text1[i - 1] == text2[j - 1]){dp[i, j] = dp[i - 1, j - 1] + 1;}else{dp[i, j] = Math.Max(dp[i - 1, j], dp[i, j - 1]);}}}return dp[text1.Length, text2.Length];}public static void Main(){string text1 = "ABCBDAB";string text2 = "BDCAB";int length = FindLongestCommonSubsequence(text1, text2);Console.WriteLine("最长公共子序列的长度是: " + length);}
}

3. 分治算法(Divide and Conquer)

分治算法是一种将一个复杂问题分解成两个或者更多的相同或相似的子问题,再将子问题分成更小的子问题,直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并。分治算法通常用于解决具有递归特点的问题,例如归并排序、快速排序等。

分治算法的主要优点是实现简单,且能够保证找到全局最优解。但缺点是递归导致的函数调用开销较大,当问题规模较大时,时间复杂度和空间复杂度较高。因此,在使用分治算法时,需要确保子问题具有独立性,且合并子问题的解能得到原问题的解。

示例:归并排序(Merge Sort)

csharp
复制
using System;

public class DivideAndConquerExample
{
public static void Merge(int[] arr, int left, int mid, int right)
{
int n1 = mid - left + 1;
int n2 = right - mid;

    int[] L = new int[n1];int[] R = new int[n2];Array.Copy(arr, left, L, 0, n1);Array.Copy(arr, mid + 1, R, 0, n2);int i = 0, j = 0;int k = left;while (i < n1 && j < n2){if (L[i] <= R[j]){arr[k] = L[i];i++;}else{arr[k] = R[j];j++;}k++;}while (i < n1){arr[k] = L[i];i++;k++;}while (j < n2){arr[k] = R[j];j++;k++;}
}public static void MergeSort(int[] arr, int left, int right)
{if (left < right){int mid = left + (right - left) / 2;MergeSort(arr, left, mid);MergeSort(arr, mid + 1, right);Merge(arr, left, mid, right);}
}public static void Main()
{int[] arr = { 12, 11, 13, 5, 6, 7 };int n = arr.Length;MergeSort(arr, 0, n - 1);Console.WriteLine("排序后的数组: ");for (int i = 0; i < n; i++){Console.Write(arr[i] + " ");}
}

}

4. 回溯算法(Backtracking)

回溯算法是一种通过尝试分步选择的方法去解决问题的策略。它在每一步尝试所有可能的选择,如果当前选择导致无法达到期望的结果,则回溯到上一步,尝试另一个选项。回溯算法通常用于解决具有约束条件的组合问题,例如旅行商问题、0-1背包问题等。

回溯算法的主要优点是能够找到所有可能的解。但缺点是计算量较大,当问题规模较大时,时间复杂度和空间复杂度较高。因此,在使用回溯算法时,需要合理设计搜索状态空间和剪枝策略,以减少不必要的计算。

示例:八皇后问题(Eight Queens Problem)

using System;public class BacktrackingExample
{public static bool PlaceQueen(int[,] board, int row){for (int col = 0; col < board.GetLength(1); col++){if (IsSafe(board, row, col)){board[row, col] = 1;if (row == board.GetLength(0) - 1){return true;}if (PlaceQueen(board, row + 1)){return true;}board[row, col] = 0;}}return false;}public static bool IsSafe(int[,] board, int row, int col){for (int i = 0; i < row; i++){if (board[i, col] == 1){return false;}}for (int i = row, j = col; i >= 0 && j >= 0; i--, j--){if (board[i, j] == 1){return false;}}for (int i = row, j = col; i < board.GetLength(0) && j < board.GetLength(1); i++, j++){if (board[i, j] == 1){return false;}}return true;}public static void Main(){int n = 8;int[,] board = new int[n, n];if (PlaceQueen(board, 0)){Console.WriteLine("八皇后问题的解决方案:");for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){Console.Write(board[i, j] + " ");}Console.WriteLine();}}else{Console.WriteLine("八皇后问题没有解决方案。");}}
}

以上是四种常用启发式算法的简要介绍和示例。每种算法都有其适用的场景和特点,根据具体问题的性质选择合适的算法可以提高解决问题的效率。

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

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

相关文章

P8602蓝桥杯大臣找路

很久以前&#xff0c;T 王国空前繁荣。为了更好地管理国家&#xff0c;王国修建了大量的快速路&#xff0c;用于连接首都和王国内的各大城市。 为节省经费&#xff0c;T 国的大臣们经过思考&#xff0c;制定了一套优秀的修建方案&#xff0c;使得任何一个大城市都能从首都直接…

政安晨:【深度学习神经网络基础】(八)—— 神经网络评估回归与模拟退火训练

目录 简述 评估回归 模拟退火训练 政安晨的个人主页&#xff1a;政安晨 欢迎 &#x1f44d;点赞✍评论⭐收藏 收录专栏: 政安晨的机器学习笔记 希望政安晨的博客能够对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff01; 简述 深度学习神经网…

安全调用(?.) Elvis运算符(?:)

安全调用&#xff08;?.&#xff09; 安全调用运算符允许开发者在可能为空的对象上安全地访问属性或调用方法。如果对象不为空&#xff0c;操作就会被执行&#xff1b;如果对象为空&#xff0c;则跳过操作&#xff0c;并返回null而不是抛出NullPointerException。 val lengt…

【算法刷题 day27】Leetcode:39. 组合总和 40.组合总和II 131.分割回文串

39. 组合总和 文档链接&#xff1a;[代码随想录] 题目链接&#xff1a;39. 组合总和 题目&#xff1a; 给你一个 无重复元素 的整数数组 candidates 和一个目标整数 target &#xff0c;找出 candidates 中可以使数字和为目标数 target 的 所有 不同组合 &#xff0c;并以列表形…

全排列问题

日升时奋斗&#xff0c;日落时自省 目录 1、全排列 2、全排列II 3、子集 4、组合 1、全排列 首先要了解全排列是怎么样的 例如:数组[1,2,3]的全排列&#xff08;全排列就是不同顺序排列方式&#xff09; 例子所有的排列方式如&#xff1a;[1,2,3],[1,3,2],[2,1,3],[2,3…

大话设计模式之享元模式

享元模式是一种结构型设计模式&#xff0c;旨在有效地支持大量细粒度的对象共享&#xff0c;从而减少内存消耗和提高性能。 在享元模式中&#xff0c;对象分为两种&#xff1a;内部状态&#xff08;Intrinsic State&#xff09;和外部状态&#xff08;Extrinsic State&#xf…

初级软件测试常见问题

1.JMeter &#xff08;1&#xff09;在http请求的时候&#xff0c;消息体数据中的数据需要用{}和“”标记起来&#xff0c;变量要用${}括起来。 &#xff08;2&#xff09;在响应断言的时候&#xff0c;要根据测试模式输出的内容来改变测试字段&#xff0c;假如输出错误可以把…

vscode 调试debug gdb vector string等STL容器,指定长度

主要展示2个调试信息&#xff1a; 1. 数组 *tr20&#xff0c;指tr数组的前20个元素 2.Vector *(int(*)[5])a ,指a容器前5个元素&#xff0c;也可以解决1的问题 二维数组 -exec p/d b也可以 附&#xff1a;命令参考 gdb 调试常用命令 - 红旗kernel - 博客园 (cnblogs.com) GD…

书生·浦语大模型开源体系(五)笔记

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…

定时器、PWM定时器、UART串口通信

我要成为嵌入式高手之4月15日ARM第八天&#xff01;&#xff01; ———————————————————————————— 定时器 S3C2440A 有 5 个 16 位定时器。其中定时器 0、1、2 和 3 具有脉宽调制&#xff08;PWM&#xff09;功能。定时器 4 是一个无 输出引脚的内部…

部署项目的时候的一些错误

项目打jar包&#xff0c;找不到资源&#xff0c;连接不上数据库 项目打包后无法运行 直接在idea运行可以 解决方法&#xff1a;pom文件中增加&#xff08;配置文件如果是yml&#xff0c;写yml&#xff09; <resources><resource><directory>src/main/java&…

MySQL—MySQL架构

MySQL—MySQL架构 MySQL逻辑架构图如下&#xff1a; Connectors连接器:负责跟客户端建立连接&#xff1b;Management Serveices & Utilities系统管理和控制工具&#xff1b;Connection Pool连接池:管理用户连接&#xff0c;监听并接收连接的请求&#xff0c;转发所有连接的…

使用Scrapy选择器提取豆瓣电影信息,并用正则表达式从介绍详情中获取指定信息

本文同步更新于博主个人博客&#xff1a;blog.buzzchat.top 一、Scrapy框架 1. 介绍 在当今数字化的时代&#xff0c;数据是一种宝贵的资源&#xff0c;而网络爬虫&#xff08;Web Scraping&#xff09;则是获取网络数据的重要工具之一。而在 Python 生态系统中&#xff0c;S…

深度学习遇到的疑问记录

深度学习遇到的疑问记录 1. Softmax和Sigmoid之间的区别和联系2. 语音唤醒模型为什么经常选择使用CNN而不是RNN3. 什么是前馈神经网络4. 前馈神经网络包含cnn吗4. 简单介绍一些LSTM网络结构5. 什么是GRU 1. Softmax和Sigmoid之间的区别和联系 Softmax和Sigmoid都是神经网络中常…

hadoop编程之部门工资求和

数据集展示 7369SMITHCLERK79021980/12/17800207499ALLENSALESMAN76981981/2/201600300307521WARDSALESMAN76981981/2/221250500307566JONESMANAGER78391981/4/22975207654MARTINSALESMAN76981981/9/2812501400307698BLAKEMANAGER78391981/5/12850307782CLARKMANAGER78391981/…

Rust语言入门第五篇-数据类型

文章目录 数据类型1.标量类型1. 整数类型2.浮点数类型f32 和 f64示例代码注意事项 3.布尔类型4.字符类型 2.复合类型整数类型技术细节1. 检查溢出&#xff08;Checking Overflow&#xff09;2. 溢出时 panic&#xff08;Panic on Overflow&#xff09;3. 使用 Wrapping 模式&am…

hadoop编程之工资序列化排序

数据集展示 7369SMITHCLERK79021980/12/17800207499ALLENSALESMAN76981981/2/201600300307521WARDSALESMAN76981981/2/221250500307566JONESMANAGER78391981/4/22975207654MARTINSALESMAN76981981/9/2812501400307698BLAKEMANAGER78391981/5/12850307782CLARKMANAGER78391981/…

【C语言基础】:预处理详解(二)

文章目录 一、宏和函数的对比二、#和##运算符2.1 #运算符2.2 ##运算符 三、#undef四、命令行定义五、条件编译六、头文件的包含1. 头文件包含的方式2. 嵌套文件包含 上期回顾&#xff1a; 【C语言基础】&#xff1a;预处理详解(一) 一、宏和函数的对比 宏通常被应有于执行简单…

蓝桥杯-回文日期

回文日期&#xff1a;回文日期 //找到一个年份&#xff0c;倒序输出符合条件就行 #include<iostream> using namespace std; string tmp1,tmp2; int mx0; int flag11,flag20; bool is(int n){int date[20]{0,31,28,31,30,31,30,31,31,30,31,30,31};int dayn%100;int m…

【Quartz】Quartz表的SQL语句,版本2.3.0

下载地址 https://www.quartz-scheduler.org/downloads/files/quartz-2.3.0-distribution.tar.gz 文件所在路径 quartz-2.3.0-SNAPSHOT\src\org\quartz\impl\jdbcjobstore 一般使用MySQL的场景下关注tables_mysql.sql和tables_mysql_innodb.sql两个文件即可&#xff0c;笔者…