目錄
🍉概述
🍉 步驟
🍉 優缺點
🍈優點
🍈缺點
🍈應對策略
🍉示例
🍈旅行商問題
🍍步驟
🍍分解代碼
🍎包含頭文件
🍎定義函數(計算總距離)
🍎生成初始解
🍎生成鄰域解
🍎爬山算法
🍎主函數
🍍完整代碼
🍍代碼說明
🍍代碼分析
🍎時間復雜度分析
🍎空間復雜度分析
🍉結論
🍉概述
- 爬山算法(Hill Climbing Algorithm)是一種啟發式搜索算法,用于尋找給定問題的局部最優解。它是一種貪婪算法,每一步都選擇當前狀態中最佳的鄰居狀態,直到不能再找到更好的鄰居為止。
- 爬山算法通常用于優化問題中,如函數優化、路徑規劃等。盡管它可能會陷入局部最優解,但其簡單性和有效性使得它在許多實際應用中得到廣泛使用。
🍉 步驟
初始化:隨機選擇或者指定一個初始解作為起點。
評估:計算當前位置的解的價值或者成本,即目標函數的值。
鄰域搜索:尋找當前位置的鄰居解,即通過微小的變化產生相鄰的解。
移動:選擇最好的鄰居解作為下一步的位置。如果該鄰居解比當前解更優,則移動到該鄰居解;否則算法停止,當前解即為局部最優解。
終止條件:根據特定條件判斷是否達到停止搜索的條件,例如達到一定迭代次數、解的改進不顯著等。
🍉 優缺點
🍈優點
簡單易懂:爬山算法的概念和實現都非常簡單,易于理解和編碼。這使得它成為許多初學者和快速原型開發的首選算法。
計算效率高:由于每一步只需要比較有限的鄰域解,爬山算法的單步計算速度較快,在搜索空間較小或結構較簡單的問題上能快速收斂。
局部優化:在許多實際問題中,找到一個局部最優解已經可以滿足需求。爬山算法擅長在局部區域內找到較好的解。
🍈缺點
容易陷入局部最優解:爬山算法很容易陷入局部最優解,而無法找到全局最優解。尤其是在復雜或具有多個局部最優解的問題上,這一缺點尤為明顯。
缺乏全局視野:爬山算法沒有跳出局部最優解的機制,因此在具有復雜地形(多峰)的搜索空間中,性能較差。
對初始解敏感:算法的最終結果依賴于初始解的選擇,不同的初始解可能會導致不同的局部最優解。
不適用于大規模問題:對于大規模問題(如具有大量變量或復雜約束的優化問題),爬山算法的效果不佳,因為它無法有效探索大規模搜索空間。
🍈應對策略
為克服爬山算法的這些缺點,通常可以結合其他優化策略,如:
- 模擬退火算法(Simulated Annealing):通過允許偶爾接受較差的解來跳出局部最優解。
- 遺傳算法(Genetic Algorithm):通過模擬自然選擇和遺傳變異的過程來探索更廣闊的解空間。
- 禁忌搜索(Tabu Search):通過記錄最近訪問過的解,避免重復搜索和陷入局部最優。
- 多次運行:多次運行爬山算法,每次使用不同的隨機初始解,從而增加找到全局最優解的概率。
🍉示例
🍈旅行商問題
旅行商問題(TSP)是一個經典的組合優化問題,目標是在一組城市中找到最短的閉環路徑,使得每個城市僅訪問一次并最終回到起點城市。
🍍步驟
- 初始化:隨機生成一個可行解,即生成一個隨機的城市訪問順序。
- 評估:計算當前路徑的總距離。
- 鄰域搜索:生成當前路徑的鄰域解。鄰域可以通過交換路徑中的兩個城市順序、逆序某段路徑等方式生成。
- 移動:選擇鄰域中距離最短的路徑作為新的當前路徑,如果新路徑比當前路徑更優,則移動到新路徑;否則,算法停止。
- 終止條件:達到指定的迭代次數或沒有更優的鄰域解。
🍍分解代碼
🍎包含頭文件
#include <iostream>
#include <vector>
#include <algorithm>
#include <cstdlib>
#include <ctime>
#include <climits>using namespace std;
🍎定義函數(計算總距離)
double calculateTotalDistance(const vector<int>& solution, const vector<vector<double>>& distanceMatrix) {double totalDistance = 0;for (size_t i = 0; i < solution.size(); ++i) {totalDistance += distanceMatrix[solution[i]][solution[(i + 1) % solution.size()]];}return totalDistance;
}
🍎生成初始解
vector<int> initialSolution(int numCities) {vector<int> solution(numCities);for (int i = 0; i < numCities; ++i) {solution[i] = i;}random_shuffle(solution.begin(), solution.end());return solution;
}
🍎生成鄰域解
vector<vector<int>> getNeighbors(const vector<int>& solution) {vector<vector<int>> neighbors;for (size_t i = 0; i < solution.size(); ++i) {for (size_t j = i + 1; j < solution.size(); ++j) {vector<int> neighbor = solution;swap(neighbor[i], neighbor[j]);neighbors.push_back(neighbor);}}return neighbors;
}
🍎爬山算法
pair<vector<int>, double> hillClimbing(const vector<vector<double>>& distanceMatrix, int maxIterations) {int numCities = distanceMatrix.size();vector<int> currentSolution = initialSolution(numCities);double currentDistance = calculateTotalDistance(currentSolution, distanceMatrix);for (int iteration = 0; iteration < maxIterations; ++iteration) {vector<vector<int>> neighbors = getNeighbors(currentSolution);vector<int> newSolution = currentSolution;double newDistance = currentDistance;for (const auto& neighbor : neighbors) {double neighborDistance = calculateTotalDistance(neighbor, distanceMatrix);if (neighborDistance < newDistance) {newSolution = neighbor;newDistance = neighborDistance;}}if (newDistance < currentDistance) {currentSolution = newSolution;currentDistance = newDistance;} else {break;}}return {currentSolution, currentDistance};
}
🍎主函數
int main() {srand(time(0));// 示例:城市間的距離矩陣vector<vector<double>> distanceMatrix = {{0, 29, 20, 21},{29, 0, 15, 17},{20, 15, 0, 28},{21, 17, 28, 0}};int maxIterations = 1000;auto result = hillClimbing(distanceMatrix, maxIterations);vector<int> optimalPath = result.first;double optimalDistance = result.second;cout << "Optimal path: ";for (int city : optimalPath) {cout << city << " ";}cout << endl;cout << "Total distance: " << optimalDistance << endl;return 0;
}
🍍完整代碼
#include <iostream>
#include <vector>
#include <algorithm>
#include <cstdlib>
#include <ctime>
#include <climits>using namespace std;// 計算總距離函數
// 傳入參數:solution(當前城市訪問順序),distanceMatrix(城市間距離矩陣)
// 返回值:當前路徑的總距離
double calculateTotalDistance(const vector<int>& solution, const vector<vector<double>>& distanceMatrix) {double totalDistance = 0;for (size_t i = 0; i < solution.size(); ++i) {// 計算從當前城市到下一個城市的距離(環形路徑)totalDistance += distanceMatrix[solution[i]][solution[(i + 1) % solution.size()]];}return totalDistance;
}// 生成初始解函數
// 傳入參數:numCities(城市數量)
// 返回值:隨機生成的城市訪問順序
vector<int> initialSolution(int numCities) {vector<int> solution(numCities);for (int i = 0; i < numCities; ++i) {solution[i] = i;}// 隨機打亂城市訪問順序random_shuffle(solution.begin(), solution.end());return solution;
}// 生成鄰域解函數
// 傳入參數:solution(當前城市訪問順序)
// 返回值:鄰域解的集合(通過交換兩個城市順序生成)
vector<vector<int>> getNeighbors(const vector<int>& solution) {vector<vector<int>> neighbors;for (size_t i = 0; i < solution.size(); ++i) {for (size_t j = i + 1; j < solution.size(); ++j) {vector<int> neighbor = solution;// 交換兩個城市的順序swap(neighbor[i], neighbor[j]);neighbors.push_back(neighbor);}}return neighbors;
}// 爬山算法函數
// 傳入參數:distanceMatrix(城市間距離矩陣),maxIterations(最大迭代次數)
// 返回值:最優解及其對應的總距離
pair<vector<int>, double> hillClimbing(const vector<vector<double>>& distanceMatrix, int maxIterations) {int numCities = distanceMatrix.size();// 生成初始解vector<int> currentSolution = initialSolution(numCities);// 計算初始解的總距離double currentDistance = calculateTotalDistance(currentSolution, distanceMatrix);for (int iteration = 0; iteration < maxIterations; ++iteration) {// 生成當前解的鄰域解vector<vector<int>> neighbors = getNeighbors(currentSolution);vector<int> newSolution = currentSolution;double newDistance = currentDistance;// 尋找鄰域中最優的解for (const auto& neighbor : neighbors) {double neighborDistance = calculateTotalDistance(neighbor, distanceMatrix);if (neighborDistance < newDistance) {newSolution = neighbor;newDistance = neighborDistance;}}// 如果找到更優解,更新當前解if (newDistance < currentDistance) {currentSolution = newSolution;currentDistance = newDistance;} else {// 否則退出循環break;}}return {currentSolution, currentDistance};
}int main() {srand(time(0)); // 初始化隨機數種子// 示例:城市間的距離矩陣vector<vector<double>> distanceMatrix = {{0, 29, 20, 21},{29, 0, 15, 17},{20, 15, 0, 28},{21, 17, 28, 0}};int maxIterations = 1000; // 最大迭代次數auto result = hillClimbing(distanceMatrix, maxIterations); // 調用爬山算法vector<int> optimalPath = result.first; // 最優路徑double optimalDistance = result.second; // 最優路徑的總距離// 輸出最優路徑cout << "Optimal path: ";for (int city : optimalPath) {cout << city << " ";}cout << endl;// 輸出最優路徑的總距離cout << "Total distance: " << optimalDistance << endl;return 0;
}
🍍代碼說明
- 頭文件:包含必要的頭文件。
- 計算總距離:
calculateTotalDistance
函數計算路徑的總距離。- 生成初始解:
initialSolution
函數生成一個隨機的城市訪問順序。- 生成鄰域解:
getNeighbors
函數生成當前解的鄰域解。- 爬山算法:
hillClimbing
函數實現爬山算法,尋找旅行商問題的最優解。- 主函數:
main
函數中定義了距離矩陣,并調用爬山算法找到最優路徑和距離。
🍍代碼分析
🍎時間復雜度分析
計算總距離函數 calculateTotalDistance
double calculateTotalDistance(const vector<int>& solution, const vector<vector<double>>& distanceMatrix)
- 該函數需要遍歷路徑中的所有城市,因此其時間復雜度為 O(n),其中 n 是城市的數量。
生成初始解函數 initialSolution
vector<int> initialSolution(int numCities)
- 該函數需要初始化一個大小為 n 的向量,并對其進行隨機打亂。初始化向量的時間復雜度為 O(n),隨機打亂的時間復雜度為 O(n),因此總的時間復雜度為O(n)。
生成鄰域解函數 getNeighbors
vector<vector<int>> getNeighbors(const vector<int>& solution)
- 該函數生成所有可能的鄰域解,對于每對城市交換一次,生成一個新的鄰域解。因此,其時間復雜度為O(n^2)。
爬山算法函數 hillClimbing
pair<vector<int>, double> hillClimbing(const vector<vector<double>>& distanceMatrix, int maxIterations)
該函數的主要步驟包括:
- 生成初始解,時間復雜度為 O(n)。
- 在每次迭代中:
- 生成鄰域解,時間復雜度為 O(n^2)。
- 遍歷鄰域解并計算每個解的總距離,時間復雜度為 O(n^3)(每個鄰域解計算距離為 O(n),有 O(n^2) 個鄰域解)。
- 迭代最多
maxIterations
次。因此,爬山算法的總時間復雜度為 O(maxIterations×n^3)。
🍎空間復雜度分析
計算總距離函數 calculateTotalDistance
- 該函數使用常量空間,空間復雜度為 O(1)。
生成初始解函數 initialSolution
- 該函數返回一個大小為 n 的向量,因此空間復雜度為 O(n)。
生成鄰域解函數 getNeighbors
- 該函數生成所有鄰域解,最多有 O(n^2) 個鄰域解,每個鄰域解的大小為 n。因此,空間復雜度為 O(n^3)。
爬山算法函數 hillClimbing
- 該函數的主要空間開銷來自:
- 因此,爬山算法的總空間復雜度為 O(n^3)。
- 初始解和當前解的存儲,空間復雜度為 O(n)。
- 鄰域解的存儲,空間復雜度為 O(n^3)。
這些復雜度分析表明,爬山算法在解決旅行商問題時,對于較大的城市數量 �n 以及較高的迭代次數 maxIterations
,可能會遇到計算和存儲資源的限制。
🍉結論
- 爬山算法作為一種簡單、快速的局部搜索優化算法,在解決一些中小規模問題時非常有效。然而,由于其容易陷入局部最優解且缺乏全局視野,在復雜或大規模優化問題上,需要結合其他優化策略以提升性能和解決問題的能力。
希望這些能對剛學習算法的同學們提供些幫助哦!!!