六.贪心算法

1.贪心算法

在对问题求解时,总是做出在当前看来是最好的选择.
也就是说,不从整体最优上加以考虑,而是局部最优解

贪心不保证最优解,但在某些问题上,贪心算法的解就是最优解

2. 找零问题

t = [100, 50, 20, 5, 1]
def change(t, n):
    m = [0 for _ in range(len(t))]
    for i, money in enumerate(t):
        m[i] = n // money
        n = n % money
    return m, n

print(change(t, 376))

3.分数背包问题

"""
0-1背包:对于一个商品,小偷要么完整的拿走,要么留下(如金条)
"""
"""
分数背包:对于一个商品,小偷可以拿走其中任意一部分(如金砂)
"""
"""
对于0-1背包和分数背包,贪心算法是否能得到最优解?
    分数背包,贪心算法可以得到最优解
    0-1背包,贪心算法不一定是最优解(最后可能背包剩下好多容量)
"""
goods = [(60,10),(100, 20),(120,30)]  # 每个商品的价格和重量
goods.sort(key=lambda x: x[0]/x[1], reverse=True)

def fraction_backpack(goods, w):
    """分数背包"""
    m = [0 for _ in range(len(goods))]
    total_v = 0
    for i, (price, weight) in enumerate(goods):
        if w == 0:
            break
        m[i] = 1 if w >= goods[i][1] else w/goods[i][1]
        w -= m[i]*goods[i][1]
        total_v += goods[i][0]*m[i]
    return m, total_v

print(fraction_backpack(goods, 50))

4.拼接最大数字问题

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# author: Xiang Hai
# wechat: xiaoyou42952
"""
拼接最大数字问题
有n个非负整数,将其按照字符串拼接的方式,拼接为一个整数,
如何拼接最大?
例如:32, 94, 128, 1286, 6, 71 可以拼接为的最大整数为:
    94716321286128
"""
from functools import cmp_to_key
li = [32, 94, 128, 1286, 6, 71]


def xy_cmp(x,y):
    if x+y < y+x:
        return 1
    elif x+y > y+x:
        return -1
    else:
        return 0


def number_join(li):
    li = list(map(str, li))
    li.sort(key=cmp_to_key(xy_cmp))
    return "".join(li)


print(number_join(li))

5.活动选择问题

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# author: Xiang Hai
# wechat: xiaoyou42952
"""
假设有n个活动,这些活动要占用同一个场地,在某时刻只能供一个活动使用
每个活动都有开始时间si和结束时间fi,表示活动在[si,fi)区间占用场地
问:安排哪些活动能够使得该场地举办的活动个数最多?

贪心:最先结束的活动一定是最优解的一部分
"""

activities = [(1,4),(3,5),(0,6),(5,7),(3,9),(5,9),(6,10),(8,11),(8,12),(2,14),(12,16)]
# 保证活动是按照结束时间排好序的
activities.sort(key=lambda x: x[1])


def activity_selection(a):
    res = [a[0]]
    for i in range(1, len(a)):
        if a[i][0] >= res[-1][1]:  # 当前活动的开始时间大于等于最后一个入选活动的结束时间
            res.append(a[i])
    return res


print(activity_selection(activities))

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值