为了账号安全,请及时绑定邮箱和手机立即绑定

从零开始学习贪心算法

概述

本文详细介绍了贪心算法的基本概念、特点和应用场景,解释了其核心思想和实现步骤,并通过几个实例展示了贪心算法的实际应用。文中还分析了贪心算法的优势与局限性,提供了相关的学习资源。

贪心算法简介

贪心算法是一种常用的算法设计方法,主要用于解决优化问题。其基本思想是在每一步选择中都采取当前状态下最优的选择,即局部最优解,从而希望最终得到的结果也是全局最优解。

贪心算法的基本概念

贪心算法通过一系列局部最优解逐步构建全局最优解。具体而言,贪心算法从问题的初始状态出发,每一步都根据某一局部最优原则做出决策,逐步向最终状态推进。最终结果是所有局部最优解的组合。

贪心算法的特点和适用场景

特点

  1. 局部最优解:每一步选择当前状态下的最优解。
  2. 贪心选择性质:每个阶段的选择都是独立的,仅依赖于当前状态的信息。
  3. 贪心算法的时间复杂度:通常较低,因为每次选择只考虑局部最优。
  4. 易于实现:实现较为简单,通常只需要几行代码。

适用场景

  1. 问题具有贪心选择性质:每一步的选择都是独立的,且依赖于当前状态。
  2. 问题具有最优子结构:子问题的最优解可以组合成全局最优解。

贪心算法与动态规划的区别

动态规划和贪心算法都是优化问题的常用方法,但它们的策略和适用场景有所不同。

  • 动态规划:通常需要考虑所有可能的子问题,并通过子问题的解来构建全局最优解。动态规划通常使用递归或迭代的方式,且需要记录子问题的解以避免重复计算。
  • 贪心算法:每次只考虑局部最优解,并期望这些局部最优解能够组合成全局最优解。贪心算法通常不涉及子问题的解,而是直接根据当前状态做出决策。

贪心算法的核心思想

贪心算法的核心思想是在每一步选择中都采取当前状态下最优的选择,即局部最优解,并期望这些局部最优解能够组合成全局最优解。

局部最优解与全局最优解

局部最优解是指在当前状态下做出的最优选择。贪心算法的目标是通过一系列局部最优解逐步构建全局最优解。然而,并不是所有问题都能通过局部最优解来保证全局最优解,例如著名的“旅行商问题”就无法通过贪心算法得到全局最优解。

贪心选择性质和最优子结构

  • 贪心选择性质:每一步选择都是当前状态下的最优选择,不依赖于未来的选择。
  • 最优子结构:每个子问题的最优解都能够组合成全局最优解。

实例解析:最小路径问题

最小路径问题是贪心算法的一个应用实例。假设有一组点,每对点之间都有一条路径,目标是找到从起点到终点的最短路径。这里我们使用Prim算法来实现最小路径问题。

import heapq

def prim(graph, start):
    n = len(graph)
    visited = [False] * n
    visited[start] = True
    edges = [(graph[start][i], start, i) for i in range(n)]
    heapq.heapify(edges)
    mst = []

    while edges:
        cost, u, v = heapq.heappop(edges)
        if not visited[v]:
            visited[v] = True
            mst.append((u, v, cost))
            for i in range(n):
                if not visited[i] and graph[v][i] > 0:
                    heapq.heappush(edges, (graph[v][i], v, i))
    return mst

# 测试数据
graph = [
    [0, 4, 0, 0, 0, 0, 0, 8, 0],
    [4, 0, 8, 0, 0, 0, 0, 11, 0],
    [0, 8, 0, 7, 0, 4, 0, 0, 2],
    [0, 0, 7, 0, 9, 14, 0, 0, 0],
    [0, 0, 0, 9, 0, 10, 0, 0, 0],
    [0, 0, 4, 14, 10, 0, 2, 0, 0],
    [0, 0, 0, 0, 0, 2, 0, 1, 6],
    [8, 11, 0, 0, 0, 0, 1, 0, 7],
    [0, 0, 2, 0, 0, 0, 6, 7, 0]
]
print(prim(graph, 0))
``

这段代码实现了一个使用Prim算法求解最小路径问题的贪心算法。Prim算法通过每次选择当前状态下的最优解来逐步构建全局最优解。

### 贪心算法的实现步骤

贪心算法的实现通常包括以下几个步骤:

1. **确定问题的贪心准则**:明确每一步选择时的目标是什么。
2. **设计贪心算法的实现步骤**:根据贪心准则设计具体的实现步骤。
3. **实现代码**:编写代码实现贪心算法。

#### 简单示例:硬币找零问题

硬币找零问题是通过最小数量的硬币来找零钱。假设硬币面值为1元、2元、5元,当我们需要找零6元时,可以通过选择一枚1元和一枚5元,或者两枚2元和一枚2元来实现。

```python
def coin_change(coins, amount):
    coins.sort(reverse=True)
    result = []
    for coin in coins:
        while amount >= coin:
            amount -= coin
            result.append(coin)
    if amount == 0:
        return result
    return "Cannot make exact amount with given coins."

coins = [1, 2, 5]
amount = 11
print(coin_change(coins, amount))

这段代码实现了一个使用贪心算法实现的硬币找零问题。硬币找零问题的目标是使用最少数量的硬币来找零给定的金额,每次选择当前最优解。

常见的贪心算法应用

贪心算法在许多实际问题中都有应用,如活动选择问题、集合覆盖问题、图的最小生成树问题等。

活动选择问题

活动选择问题是指在给定一系列活动和每个活动的开始和结束时间的情况下,选择尽可能多的不重叠的活动。

def activity_selector(start, finish):
    n = len(start)
    activities = sorted(range(n), key=lambda i: finish[i])
    count = 1
    i = 0
    j = 1
    while j < n:
        if start[j] >= finish[i]:
            count += 1
            i = j
        j += 1
    return count

start = [1, 3, 0, 5, 3, 5, 6, 8, 8, 2, 12]
finish = [4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14]
print(activity_selector(start, finish))

这段代码实现了一个活动选择问题的贪心算法。活动选择问题的目标是选择尽可能多的不重叠活动,每次选择当前最早结束的活动。

集合覆盖问题

集合覆盖问题是指从给定的一组集合中选择最少数量的集合,使得这些集合的并集覆盖所有元素。

def set_cover(universe, subsets):
    selected_subsets = set()
    covered_elements = set()
    while covered_elements != universe:
        subset_ratios = [(len(subset - covered_elements), subset) for subset in subsets]
        best_subset = max(subset_ratios)[1]
        selected_subsets.add(best_subset)
        covered_elements |= best_subset
    return selected_subsets

universe = set(range(1, 11))
subsets = [set([2, 6, 8]), set([1, 3, 7, 9, 10]), set([1, 2, 4, 5]), set([2, 3, 4, 5, 6, 7]), set([3, 4, 8]), set([4, 5, 6, 8, 9, 10])]
print(set_cover(universe, subsets))

这段代码实现了一个集合覆盖问题的贪心算法。集合覆盖问题的目标是选择最少数量的集合,使得这些集合的并集覆盖所有元素,每次选择覆盖元素最多的集合。

图的最小生成树问题

图的最小生成树问题是指在给定一个加权无向图中,找到一个子图作为该图的一棵最小生成树,这棵最小生成树的代价(所有边的权重之和)最小。

class Graph:
    def __init__(self, vertices):
        self.V = vertices
        self.graph = []

    def add_edge(self, u, v, w):
        self.graph.append([u, v, w])

    def find(self, parent, i):
        if parent[i] == i:
            return i
        return self.find(parent, parent[i])

    def union(self, parent, rank, x, y):
        xroot = self.find(parent, x)
        yroot = self.find(parent, y)
        if rank[xroot] < rank[yroot]:
            parent[xroot] = yroot
        elif rank[xroot] > rank[yroot]:
            parent[yroot] = xroot
        else:
            parent[yroot] = xroot
            rank[xroot] += 1

    def prim(self):
        result = []
        i = 0
        e = 0
        graph = sorted(self.graph, key=lambda item: item[2])
        parent = []
        rank = []
        for node in range(self.V):
            parent.append(node)
            rank.append(0)
        while e < self.V - 1 and i < len(graph):
            u, v, w = graph[i]
            i += 1
            x = self.find(parent, u)
            y = self.find(parent, v)
            if x != y:
                e += 1
                result.append([u, v, w])
                self.union(parent, rank, x, y)
        return result

g = Graph(4)
g.add_edge(0, 1, 10)
g.add_edge(0, 2, 6)
g.add_edge(0, 3, 5)
g.add_edge(1, 3, 15)
g.add_edge(2, 3, 4)
print(g.prim())

这段代码实现了一个使用Prim算法求解最小生成树问题的贪心算法。Prim算法通过每次选择当前最小的边来逐步构建最小生成树。

贪心算法的优缺点分析

贪心算法的优势

  1. 简单:贪心算法通常实现较为简单,易于理解和实现。
  2. 高效:贪心算法通常具有较低的时间复杂度,能够快速解决问题。
  3. 局部最优解:每一步都选择当前状态下的最优解,容易理解和分析。

贪心算法的局限性

  1. 局部最优不等于全局最优:并非所有问题都能通过局部最优解来保证全局最优解。
  2. 依赖于特定条件:贪心算法适用的场景有限,需要满足一定的条件才能保证正确性。
  3. 难以证明最优性:有时候难以证明贪心算法的选择是全局最优的。

如何应对贪心算法的不足

  1. 证明正确性:通过数学证明或者其他方法证明贪心算法的选择是正确的。
  2. 增加约束条件:在特定条件下应用贪心算法,确保满足条件时能够得到全局最优解。
  3. 结合其他算法:结合动态规划、回溯等其他算法,提高算法的鲁棒性和正确性。

贪心算法实战练习

贪心算法的实战练习可以帮助我们更好地理解和掌握贪心算法的应用。

练习题解析

题目:假设你有一堆硬币,每种硬币都有无限数量,面值分别为1元、2元、5元。给定一个总金额,找出组成该金额所需的最少硬币数。

def coin_change(coins, amount):
    coins.sort(reverse=True)
    result = []
    for coin in coins:
        while amount >= coin:
            amount -= coin
            result.append(coin)
    if amount == 0:
        return result
    return "Cannot make exact amount with given coins."

coins = [1, 2, 5]
amount = 11
print(coin_change(coins, amount))

这段代码实现了一个使用贪心算法解决硬币找零问题的算法。通过每次选择当前面值最大的硬币来逐步构建最少硬币数。

实战案例分享

案例:假设有一个仓库,仓库里有不同大小的箱子,箱子可以被堆叠。你需要找到一个方法来最大化堆叠箱子的数量,使得每个箱子都放在一个比它大的箱子上,或者放在地面上。

def max_stack_boxes(boxes):
    boxes.sort()
    n = len(boxes)
    dp = [1] * n
    for i in range(1, n):
        for j in range(i):
            if boxes[i][0] > boxes[j][0] and boxes[i][1] > boxes[j][1]:
                dp[i] = max(dp[i], dp[j] + 1)
    return max(dp)

boxes = [(1, 2), (2, 3), (3, 4), (4, 5)]
print(max_stack_boxes(boxes))

这段代码实现了一个使用动态规划解决堆叠箱子问题的算法。堆叠箱子问题的目标是找到一种方法来最大化堆叠箱子的数量,使得每个箱子都放在一个比它大的箱子上,或者放在地面上,每次选择当前最优解。

总结与心得

贪心算法是一种简单而高效的方法,适用于许多优化问题。通过局部最优解逐步构建全局最优解,使得贪心算法具有较低的时间复杂度和较好的易于理解和实现的特点。然而,贪心算法也有其局限性,需要满足特定条件才能保证全局最优解。通过实践和练习,可以更好地理解和掌握贪心算法的应用。推荐学习网站如慕课网提供丰富的贪心算法和编程课程,可以帮助进一步提升编程技能。

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消