动态规划详解Python

news2024/9/21 16:39:35

动态规划

动态规划(Dynamic Programming)是一种用于解决复杂问题的算法设计方法。它通常用于优化问题,其中问题可以被分解成一系列重叠子问题,通过存储并重复使用已经解决过的子问题的解,可以避免重复计算,从而提高算法的效率。

动态规划的基本思想是将原始问题分解成若干个子问题,并逐个求解这些子问题的最优解。通过定义状态和状态转移方程,可以将问题的求解转化为一个递推过程,从而得到最优解。

动态规划算法的核心步骤通常包括以下几个方面:

  1. 定义问题的状态:将原始问题抽象为一个或多个子问题,并定义状态来表示子问题的解。
  2. 确定状态转移方程:通过分析子问题之间的关系,建立状态之间的转移方程,描述当前状态和之前状态之间的关系。
  3. 确定初始条件:确定最简单的子问题的解,即初始状态的值。
  4. 递推求解:使用状态转移方程和初始条件,逐步计算出更复杂的子问题的解,直到得到原始问题的解。
  5. 解析解:根据子问题的解,逐步还原出原始问题的解。

动态规划算法通常具有较高的时间复杂度,但通过存储已解决的子问题的解,可以大大减少重复计算,提高算法效率。它在许多领域有广泛的应用,如组合优化、图论、序列比对、路径规划等。

斐波那契数列

这里简单的解释一下斐波那契数列:F(0) = 0 , F(1) = 1
F(N) = F(N-1) + F(N-2) , N>1 数列前几项如下:
0 1 1 2 3 5 8 13 21…

递归代码和非递归代码比较

import time


def calculate_time(func):
    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        end_time = time.time()
        execution_time = end_time - start_time
        print(f"函数 {func.__name__} 的执行时间为:{execution_time} 秒")
        return result

    return wrapper


@calculate_time
def func1(n):
    return _func1(n)


# 存在大量的子问题重复计算
def _func1(n):
    if n <= 1:
        return n
    return _func1(n - 1) + _func1(n - 2)


# 把需要用到的子问题存起来
@calculate_time
def func2(n):
    f = [0, 1]
    if n > 1:
        for i in range(n - 1):
            num = f[-1] + f[-2]
            f.append(num)
    return f[n]


print(func1(36))
print(func2(36))

运行结果:
函数 func1 的执行时间为:3.150125503540039 秒
14930352
函数 func2 的执行时间为:0.0 秒
14930352

爬楼梯

问题:有一个楼梯,总共有10级台阶,每次只能走一级或者两级台阶,全部走完,有多少种走法?

找规律得到递推式:
在这里插入图片描述

def climbStairs(self, n: int) -> int:
    f = [0,1,2]
    if n > 2:
        for i in range(n-2):
            r = f[-1] + f[-2]
            f.append(r)

    return f[n]

最大回撤

问题:有一个数组,求其中两个数x,y,满足x的索引小于y的索引,使得 x-y 最大。例如 arr = [3,7,2,6,4,1,9,8,5], 最大回撤是6,对应的x=7,y=1。

初始时,设max_drop=0,peak都设为arr[0]。然后从左到右遍历数组,对于每个元素price:

  1. 如果price大于peak,更新peak为当前的price。
  2. 否则,计算当前的回撤值drop,即peak - price。
  3. 如果drop大于max_drop,更新max_drop为当前的drop。

最终,max_drop就是最大回撤的值。

# 只返回最大回撤的金额
def calculate_max_drawdown(arr):
    max_drop = 0
    peak = arr[0]

    for i in range(1, len(arr)):
        if arr[i] > peak:
            peak = arr[i]
        else:
            drop = peak - arr[i]
            if drop > max_drop:
                max_drop = drop

    return max_drop

# 示例用法
arr = [3, 7, 2, 6, 4, 1, 9, 8, 5]
max_drawdown = calculate_max_drawdown(arr)
print(max_drawdown)  # 输出: 6

要求计算出最大回撤,并且对应的返回x和y
可以使用动态规划的思想来解决这个问题。我们可以定义一个变量max_sum来跟踪当前的最大子数组和,以及一个变量current_sum来记录当前的子数组和。

初始时,将max_sum和current_sum都设为数组中的第一个元素。然后从数组的第二个元素开始遍历,对于每个元素num:

  1. 将current_sum与0比较,取其中较大的值,并将num加到current_sum中,得到新的current_sum。
  2. 将current_sum与max_sum比较,取其中较大的值,并将结果赋给max_sum。

最终,max_sum就是最大连续子数组的和。

def calculate_max_drawdown(arr):
    max_drop = 0
    peak = 0
    start_index = 0
    end_index = 0

    for i, price in enumerate(arr):
        if price > peak:
            peak = price
        else:
            drop = peak - price
            if drop > max_drop:
                max_drop = drop
                start_index = arr.index(peak)
                end_index = i

    return max_drop, start_index, end_index

# 示例用法
arr = [3, 7, 2, 6, 4, 1, 9, 8, 5]
max_drawdown, start_index, end_index = calculate_max_drawdown(arr)
print("最大回撤:", max_drawdown)
print("起始索引:", start_index)
print("结束索引:", end_index)

输出:
最大回撤: 6
起始索引: 1
结束索引: 4

最大连续子数组和

问题:给定一个数组,求其最大连续子数组的和。例如:arr = [1,5,-10,2,5,-3,2,6,-3,1]. 输出为:12。对应的连续子数组为 [2,5,-3,2,6]。

def max_subarray_sum(arr):
    max_sum = arr[0]
    current_sum = arr[0]

    for num in arr[1:]:
        current_sum = max(num, current_sum + num)
        max_sum = max(max_sum, current_sum)

    return max_sum


# 示例用法
arr = [1, 5, -10, 2, 5, -3, 2, 6, -3, 1]
max_sum = max_subarray_sum(arr)
print(max_sum)  # 输出: 12
"""同时输出对应的子数组"""
def max_subarray_sum(arr):
    max_sum = arr[0]
    current_sum = arr[0]
    start_index = 0
    end_index = 0
    temp_start_index = 0

    for i, num in enumerate(arr[1:], start=1):
        if num > current_sum + num:
            temp_start_index = i
            current_sum = num
        else:
            current_sum = current_sum + num

        if current_sum > max_sum:
            start_index = temp_start_index
            end_index = i
            max_sum = current_sum

    subarray = arr[start_index:end_index + 1]
    return max_sum, subarray


# 示例用法
arr = [1, 5, -10, 2, 5, -3, 2, 6, -3, 1]
max_sum, subarray = max_subarray_sum(arr)
print("最大连续子数组和:", max_sum)
print("最大连续子数组:", subarray)

输出:
最大连续子数组和: 12
最大连续子数组: [2, 5, -3, 2, 6]

最长不重复子串

题目形式:给定一个字符串,找出没有重复字符的最长的子串。例如输入“abcbefgf”,答案是 “cbefg”。

算法步骤如下:

  1. 定义两个指针,start 和 end,分别表示滑动窗口的起始位置和

  2. 结束位置,初始时两个指针都指向字符串的开头。

  3. 定义一个集合 seen,用于存储滑动窗口中出现过的字符。

  4. 定义两个变量 max_length 和 max_substring,分别用于记录最长子串的长度和内容,初始时都为 0。开始遍历字符串,从左到右依次移动 end 指针:
    —a.如果当前字符 s[end] 在集合 seen 中不存在,说明是一个新的字符,将其加入 seen 中,并更新 end 指针。
    —b.如果当前字符 s[end] 在集合 seen 中已经存在,说明遇到了重复字符。此时需要移动 start 指针,并更新 seen 集合,直到滑动窗口中不再有重复字符。

    —c.在每次移动 end 指针时,都需要更新 max_length 和 max_substring,以记录当前的最长子串。

  5. 遍历结束后,返回最长子串 max_substring。

def longest_unique_substring(s):
    start = 0
    end = 0
    seen = set()
    max_length = 0
    max_substring = ""

    while end < len(s):
        if s[end] not in seen:
            seen.add(s[end])
            end += 1
        else:
            if end - start > max_length:
                max_length = end - start
                max_substring = s[start:end]
            seen.remove(s[start])
            start += 1

    if end - start > max_length:
        max_substring = s[start:end]

    return max_substring

# 示例用法
s = "abcbefgf"
result = longest_unique_substring(s)
print(result)  # 输出: "cbefg"

全排列

问题:给定一个数组,找出其所有可能的排列。例如: arr = [1,1,3],输出为 [[1,1,3],[1,3,1],[3,1,1]]。

def permute_unique(nums):
    # 定义递归函数,生成给定位置上的所有可能排列
    def backtrack(start):
        # 终止条件:当遍历到数组末尾时,将当前生成的排列加入结果集
        if start == len(nums):
            permutations.append(nums[:])  # 将当前排列加入结果集
            return

        # 使用一个集合来记录已经选择过的元素,避免重复生成相同的排列
        used = set()

        # 从当前位置开始,依次尝试每个元素作为当前位置的元素
        for i in range(start, len(nums)):
            # 如果当前元素已经被选择过,则跳过
            if nums[i] in used:
                continue

            # 进行选择:将当前元素加入路径,并标记为已选择
            used.add(nums[i])
            nums[start], nums[i] = nums[i], nums[start]

            # 递归调用自身,在新的位置上生成剩余元素的所有可能排列
            backtrack(start + 1)

            # 撤销选择:将当前选择的元素从路径中移除,并将其标记为未选择,以便进行下一次选择
            nums[start], nums[i] = nums[i], nums[start]
            used.remove(nums[i])

    permutations = []  # 结果集,用于存储所有排列
    backtrack(0)  # 从位置0开始生成所有排列
    return permutations

# 示例用法
nums = [1, 1, 3]
result = permute_unique(nums)
print(result)

快速排序+二分查找

def partition(lst, left, right):
    temp = lst[left]
    while left < right:
        while left < right and lst[right] >= temp:
            right -= 1
        lst[left] = lst[right]
        while left < right and lst[left] <= temp:
            left += 1
        lst[right] = lst[left]

    lst[left] = temp

    return left


arr = [5, 2, 8, 6, 3]

print(partition(arr, 0, len(arr) - 1))


def quick_sort(lst, left, right):
    if left < right:
        mid = partition(lst, left, right)
        quick_sort(lst, left, mid)
        quick_sort(lst, mid + 1, right)


quick_sort(arr, 0, len(arr) - 1)
print(arr)


#  二分查找只适用于在有序序列中查找元素
def binary_search(lst, val):
    left = 0
    right = len(lst) - 1
    while left <= right:
        mid = (left + right) // 2
        if lst[mid] == val:
            return mid
        elif lst[mid] > val:  # 说明要查找的值在mid的左边
            right = mid - 1
        else:  # 说明要查找的值在mid的右边、
            left = mid + 1
    # 代码执行至此说明没有找到元素val,返回-1
    return -1
# 注意:mid定义在while循环里面


print(binary_search(arr, 8))

输出:
2
[2, 3, 5, 6, 8]
4

合并两个有序数组(归并排序)

题目形式:给定两个按升序排列的有序数组,将它们合并成一个新的有序数组。例如:a = [1,2,6,8], b = [2,4,7,10],输出为 arr = [1,2,2,4,6,7,8,10]

# 定义合并函数,将两个有序序列合并为一个有序序列
def merge(lst, left, mid, right):
    """
    思路:定义一个列表merged,循环比较两个有序序列的首元素大小,并放入临时空列表。
    """
    if left < right:
        merged = []
        i = left
        j = mid + 1
        while i <= mid and j <= right:
            if lst[i] < lst[j]:
                merged.append(lst[i])
                i += 1
            else:
                merged.append(lst[j])
                j += 1
        # 代码执行至此,有一个序列元素为空,另一个不为空,接下来将剩下的元素放入空列表
        while i <= mid:
            merged.append(lst[i])
            i += 1
        while j <= right:
            merged.append(lst[j])
            j += 1
        lst[left:right + 1] = merged

        return lst


def _merge_sort(lst, left, right):
    if left < right:
        mid = (left + right) // 2
        _merge_sort(lst, left, mid)
        _merge_sort(lst, mid + 1, right)
        merge(lst, left, mid, right)


@calculate_time
def merge_sort(lst):
    _merge_sort(lst, 0, len(lst) - 1)


if __name__ == '__main__':
    # 测试代码
    lst = list(range(10000))
    random.shuffle(lst)
    print(lst)
    merge_sort(lst)
    print(lst)

三数之和

def sum_of_three(arr,target):
    assert len(arr)>=3,"len(arr) should >=3!"
    arr.sort()
    ans = set()
    for k,c in enumerate(arr):
        i,j = k+1,len(arr)-1
        while i<j:
            if arr[i]+arr[j]+c <target:
                i = i+1
            elif arr[i]+arr[j]+c > target:
                j = j-1
            else:
                ans.update({(arr[k],arr[i],arr[j])})
                i = i+1
                j = j-1
    return(list(ans))
 
print(sum_of_three([-3,-1,-2,1,2,3],0))

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

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

相关文章

区块链的简单认识

比特币作为区块链的应用&#xff0c;让区块链广为人知&#xff0c;如果比特币作为第一代区块链&#xff0c;则以太坊则称为第二代区块链。我们知道&#xff0c;区块链的最主要目的就是去中心化&#xff0c;比特币则成为了decentralized currency&#xff0c;去中心化在技术上依…

跨注册中心服务同步实践

博主介绍&#xff1a;✌全网粉丝4W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战、定制、远程&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面…

助你丝滑过度到 Vue3 创建工程 ②③

作者 : SYFStrive 博客首页 : HomePage &#x1f4dc;&#xff1a; VUE3~TS &#x1f4cc;&#xff1a;个人社区&#xff08;欢迎大佬们加入&#xff09; &#x1f449;&#xff1a;社区链接&#x1f517; &#x1f4cc;&#xff1a;觉得文章不错可以点点关注 &#x1f449;…

同城跑腿独立版小程序 码科跑腿小程序 支持用户端 骑手端

是独立版哦&#xff0c;不是微擎的 搭建有点复杂&#xff0c;只要一步错就会导致骑手端来单没有声音提示. 多的也不介绍了&#xff0c;不知道的朋友可以百度一下码科跑腿就知道了&#xff01;

机器学习李宏毅学习笔记33

文章目录 前言一、神经网络压缩二、Network pruning----一种network compression技术1.移除不同单位的区别2.大乐透假说 总结 前言 神经网络压缩&#xff08;一&#xff09; 类神经网络剪枝&#xff08;pruning&#xff09; 一、神经网络压缩 简化模型&#xff0c;用比较少的…

netty_客户端和服务端,定长数据输出案例

步骤1&#xff1a;创建server import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelOption; import io.netty.channel.EventLoopGroup; import io.netty.channe…

美团小游戏守卫羊村玩法介绍和游戏漏洞

序言 这款游戏是一个解压小游戏&#xff0c;是我在闲暇时发现的&#xff0c;然后就使用它来度过无聊的碎片化时间。这是一款传统的塔防类游戏&#xff0c;建议大家可以试试&#xff0c;玩法有很多种&#xff0c;当然它的游戏优化还是不太行&#xff0c;建议多优化优化。 玩法…

十二、Jenkins构建完成发送飞书消息通知

十二、Jenkins构建完成发送飞书消息通知 1. 创建一个飞书webhook 群机器人 得到一个飞书webhook地址&#xff1a; https://open.feishu.cn/open-apis/bot/v2/hook/2d0b6357-333a-4077-9fcd-61e361a3e51e 2. send_notice.py上传到jenkins服务器目录 send_notice.py最后面 代码…

移动硬盘中安装Ubuntu 20.04系统——立省99%的问题

目录 关于我为什么要在移动硬盘中装系统 准备工作 开始安装 磁盘分区 创建虚拟机 关于我为什么要在移动硬盘中装系统 (6条消息) 笔记本安装双系统ubuntu时踩的坑——戴尔_放风筝的猪的博客-CSDN博客 准备工作 1.移动硬盘 2.Ubuntu镜像 3.VMware虚拟机 Ubuntu镜像可以从…

Linux查看版本号,lsb_releasa过时了,得用uname -a

突然想查看Linux版本号了 然后终端输入lsb_release,结果给我报错No LSB modules are available. 然后网上一查发现Linux 版本是 Ubuntu 11.1.0 或更高版本,则 lsb_release 命令可能已经被弃用。 这是由于 lsb-release 命令已经不再维护,并且由于安全问题而被移除。 因此,对于…

基于matlab使用迭代最近点算法组合多个点云以重建三维场景(附源码)

一、前言 此示例演示如何使用迭代最近点 &#xff08;ICP&#xff09; 算法组合多个点云以重建三维场景。 此示例将使用 Kinect 捕获的点云集合拼接在一起&#xff0c;以构建场景的更大三维视图。该示例将 ICP 应用于两个连续的点云。这种类型的重建可用于开发对象的 3D 模型…

关于Realvnc远程Aero效果异常解决方案

默认安装realvnc server后&#xff0c;远程时如下图所示&#xff0c;windows areo效果丢失且提示“配色方案已经更改为windows 7 basic”&#xff1b; 解决方案&#xff1a;取消勾选&#xff0c;user mirror drive to capture screen 在远程时效果恢复如下&#xff1a;

类中的函数重载

函数重载回顾 函数重载的本质为相互独立的不同函数 C 中通过函数名和函数参数确定函数调用 无法直接通过函数名得到重载函数的入口地址 函数重载必然发生在同一个作用域中 类中的成员函数可以进行重载 构造函数的重载 普通成员函数的重载 静态成员函数的重载 问题 全局…

微软发布「升级版」多模态大模型 Kosmos-2!新增局部理解能力,解锁实体级交互

夕小瑶科技说 原创 作者 | 小戏、ZenMoore 三个多月前&#xff0c;微软亚洲研究院在论文《Language Is Not All You Need: Aligning Perception with Language Models》中发布了一个强大的多模态大模型 Kosmos-1&#xff0c;成功将感知与语言对齐&#xff0c;在 ChatGPT 的多…

C#串口通信从入门到精通(26)——多个串口多个线程发送数据和接收数据

前言 我们在开发串口程序的过程中有时候会遇到多个串口,并且多个串口也需要在多个线程进行操作,本文就来讲解如何实现多个串口在多线程下的安全发送与接收。 1、操作界面与测试过程 我们首先使用虚拟串口助手虚拟COM1、COM2这一对串口;COM3、COM4这一对串口,然后使用代码…

小黑子—MySQL数据库:第二章 - 进阶篇

MySQL数据库入门2.0 MySQL进阶篇1. MySQL体系结构2. 存储引擎2.1 InnoDB 存储引擎2.2 MyISAM 存储引擎2.3 Memory 存储引擎2.4 存储引擎选择2.5 MySQL安装Linux版本 3. 索引3.1 索引结构3.1.1 B tree3.1.2 B tree3.1.3 Hash 3.2 索引分类3.2.1 思考题 3.3 索引语法3.4 SQL性能分…

docker jenkins 安装

使用 Docker 安装 Jenkins 并实现项目自动化部署-阿里云开发者社区 (aliyun.com)https://developer.aliyun.com/article/892646#slide-1运行镜像命令&#xff1a; docker run \ --name jenkins \ -p 8080:8080 \ -p 50000:50000 \ -d \ -v /home/admin/SoftWare/volume/jenkin…

CSS 内容盒子

这章比较重要&#xff0c;会不断更新❗ 文章目录 内容盒子开发者工具的使用border 边框padding 内边距margin 外边距盒子整体尺寸元素默认样式与CSS重置元素分类块级标记行级标记行内块标记 display样式内容溢出裁剪掉溢出部分滚动条 圆角边框 border-radius 内容盒子 提示&am…

云原生之使用Docker部署Firefox浏览器

云原生之使用Docker部署Firefox浏览器 一、Firefox浏览器介绍1.1 Firefox简介1.2 Firefox特点 二、本次实践介绍2.1 本地环境规划2.2 本次实践简介 三、本地环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本 四、下载Firefox镜像五、部署Firefox5…

M7615DNA M7455DNF M7675DXF安装网络打印驱动方法

注意&#xff1a;网络驱动安装首先需要把打印机接入到网络设备上&#xff0c;例如接到路由器或者交换机。 1、访问联想官方网站下载驱动或者使用机器随机自带光盘。 2、下载驱动后解压&#xff0c;如图&#xff1a; 3、运行解压后驱动包内的install文件夹里的相应打印机型号&am…