奇怪的比赛(Python,递归,状态压缩动态规划dp)

目录

  • 前言:
  • 题目:
  • 思路:
    • 递归:
      • 代码及详细注释:
    • 状态压缩dp:
      • 代码及详细注释:
  • 总结:

前言:

这道题原本是蓝桥上的题,现在搜不到了,网上关于此题的讲解更是寥寥无几,仅有的讲解也只是递归思想,python讲解和状态压缩dp的解决方法都没有,这里就带大家用状态压缩dp方法来解决此题。

题目:

大奖赛计分规则:
每位选手需要回答10个问题(其编号为1到10),越后面越有难度。答对的,当前分数翻倍;答错了,则扣掉与题号相同的分数(选手必须回答问题,不回答按错误处理)。每位选手的起步分都是10分,某获胜选手最终得分刚好是100分,请推断出哪个题目答对了,哪个题目答错了?(答对的记为1,答错的记为0,则10个题目的回答情况可用仅含1和0的串来表示)。任务是算出所有可能情况,每个答案占一行。(填空题)

输出:
0010110011
0111010000
1011010000

思路:

递归:

分析本题首先想到递归思想,从得分100逆推出初始分数为10的方案。
首先定义一个字符串

如果本题答对了,则总得分除2,字符串首位加1(因为是从第10题往前推,所以要逆序加)

如果本题答错了,则总得分加上本题的序号数,字符串首位加0

这里还有一个小细节,如果得分是奇数,则不能除2(只能答错)。

代码及详细注释:

def ScoreOut(qs, score, out):# 如果qs为0if qs == 0:# 如果score等于10if score == 10:print(out)  # 输出当前字符串outreturn out  # 返回当前字符串outelse:return ""  # 返回空字符串else:# 如果score为偶数if score % 2 == 0:# 递归调用函数,分别将score除以2并在开头添加字符"1",以及将score加上qs并在开头添加字符"O"return ScoreOut(qs - 1, score // 2, "1" + out) + ScoreOut(qs - 1, score + qs, "O" + out)else:# 如果score为奇数,只递归调用将score加上qs并在开头添加字符"O"return ScoreOut(qs - 1, score + qs, "O" + out)# 初始调用函数
ScoreOut(10, 100, "")

状态压缩dp:

什么是状态压缩

利用一串0/1数字(二进制数)来表示各个点的状态

这就要求使用状态压缩的对象的状态必须只有两种,0 或 1(如果有三种状态用三进制来表示也未尝不可)。

需要将状态数据实现为基本数据类型,比如 int,long 等,即状态压缩。比如说棋盘上的格子,放棋子或者不放;硬币的正面或反面;题目的对或错等。

状态压缩要求状态数据中的单元个数不能太大,比如用 int 来表示一个状态的时候,状态的单元个数不能超过 32(32位CPU),所以题目一般都是至少有一维的数据范围很小。

当然Python不需要考虑数的大小是否受限。

分析本题,上面用递归讲解过本题,有递归就用动态规划来解决,众所周知,递归很好理解和书写,但是递归的时间复杂度都不低,会有大量冗余计算。像本题中

如图

在这里插入图片描述
会出现大量相同得分情况,这就需要借助动态规划来处理重复计算了。

动态规划五部曲来分析

  1. 确定dp数组的定义

对于数组dp[i] :

下标表示10道题目的对错状态,dp[i]表示对应得分

下标 i 的二进制数表示对错状态,如 i = 2 时 i 的二进制数为 10 ,此时10(第1题答错,之后的提还没有做)(二进制数)的得分为多少,i = 7 时,i的二进制数为111,此时(第1题答对,第2题答对,之后的题还没有做)的得分为多少
特别注意!!! 这里首位的1(二进制后的数)没有实际含义,不表示题目对错,这个在下面会讲解

题目中有10道题,总共的有2 ** 10 - 1 种可能种情况(可能的情况用2进制计算就可以得出,如从000000000到111111111的可能数)

对于0000000000(这些都是10个数,不用挨个数了),二进制是这么表示,但10进制中,这个数就是0,

要想这个二进制数有意义,最前面就要加个1,否则无论有几个0,都表示0这个数,所以我们在定义dp数组长度的时候,不能为2 ** 10 而是为 2 ** 11(因为二进制1000000000(1后面10个0)的10进制数为 2 ** 11)

大家之前做题接触的二进制情况肯定不多,所以这里会稍微有点绕。这里主要理解前面自定义的1就好,跟补码的符号位有异曲同工之妙,只不过这里仅仅用来补位。

  1. dp数组的递推公式

状态压缩dp的递推公式还跟别的动规的递推公式不太一样,这里需要用到位运算符

这里简单讲解一下位运算符:

<< : 表示运算数的各二进制位全部左移若干位,低位补0
7 << 1 = 14, 因为7的二进制数为111,左移一位后为1110,1110的十进制数为14

右移跟左移同理。

还有python的二进制数可以用bin()函数来求,如bin(7) = 0b111,0b表示二进制,bin(7)的类型为字符串型。但是直接写0b111 就表示整型

因为第二道题的得分是由第一道题的得分算出来的,如

dp[0b1 << 1] = dp[0b1] - 1          #   dp[0b10]即dp[2]
dp[(0b1 << 1) +1] = dp[0b1] * 2      #  dp[0b11]即dp[3]

对于第cur道题,

如果答错dp[i << 1] = dp[i] - cur
如果答对dp[(i << 1) + 1] = dp[i] * 2

在推导递推公式的时候如果不太清楚,就看看dp数组的定义,切记第一个1无实义

  1. dp数组如何初始化

因为第一个1无实义,所以

dp[0b1] = 10  # dp[1]初始化得分为10
  1. dp数组遍历顺序

毫无疑问,题是从前往后答的,dp数组也是从左往右推

  1. 打印dp数组

画的有点丑,下标最好写成二进制形式。
在这里插入图片描述

求出dp数组后就把得分为100且10道题全都做了的结果输出就行。比较简单,有很多种方法,就不细说了。

代码及详细注释:

import math# 初始化一个长度为2^11的数组,初始值为0
dp = [0] * 2 ** 11
dp[1] = 10  # dp[1]初始化得分为10
# dp[0b1 <<1] = dp[0b1] - 1             dp[0b10]即dp[2]
# dp[(0b1 <<1) +1] = dp[0b1] * 2        dp[0b11]即dp[3]maxscore = 160  # 最大分数为160(因为160怎么减题目序号都到不了100,也可以设170等等)
for i in range(1, 2 ** 10):if dp[i] == -1:continueelse:cur = int(math.log2(i << 1))  # 计算准备做的题目序号,因为第一个1无实际意义,1之后的数才表示题的对错if dp[i] - cur <= 0:dp[i << 1] = -1else:dp[i << 1] = dp[i] - curif dp[i] * 2 >= maxscore:dp[(i << 1) + 1] = -1else:dp[(i << 1) + 1] = dp[i] * 2result = []
while True:if 100 in dp:index = dp.index(100)  # 找到值为100的索引dp[index] = 0if len(bin(index)) == 13:result.append(bin(index))  # 将长度为13(10道题都做完了,头三位为0b1)的二进制数添加到结果列表中else:breakfor res in result:print(res[3:])  # 输出结果列表中每个元素的第3位之后的内容

总结:

状态压缩dp还是比较难的,但把dp的定义吃透就比较好理解了。

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

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

相关文章

Python爬虫:原理与实战

引言 在当今的信息时代&#xff0c;互联网上的数据如同浩瀚的海洋&#xff0c;充满了无尽的宝藏。Python爬虫作为一种高效的数据抓取工具&#xff0c;能够帮助我们轻松地获取这些数据&#xff0c;并进行后续的分析和处理。本文将深入探讨Python爬虫的原理&#xff0c;并结合实战…

kkview远程控制: 内网远程桌面控制软件

内网远程桌面控制软件&#xff1a;高效、安全的远程管理方案 在信息技术日新月异的今天&#xff0c;内网远程桌面控制软件已成为许多企业和个人用户不可或缺的工具。这类软件允许用户通过内部网络&#xff0c;实现对其他计算机的远程访问和控制&#xff0c;从而大大提高工作效…

[Windows] Win11 常用快捷键

文章目录 &#x1f680; [Windows] Win11 常用快捷键&#x1f310; Windows 操作系统&#x1f525; Windows 11 &#x1f310; Windows 11 快捷键概览&#x1f525; 基本快捷键&#x1f525; 窗口快捷键&#x1f525; 功能快捷键 &#x1f4dd; 小结 &#x1f680; [Windows] W…

“技多不压身”是什么意思?看完这篇文章你会明白:有了手艺,走遍天下都不怕!

“技多不压身”是什么意思&#xff1f;看完这篇文章你会明白&#xff1a;有了手艺&#xff0c;走遍天下都不怕&#xff01; 咱们的老祖宗流传一句话&#xff1a;“一招鲜&#xff0c;吃遍天。”这话说得直白&#xff0c;却道出了学一门手艺或技术对于人生的重要性。“李秘书讲…

使用Loadrunner进行性能测试

一、确定性能测试的范围、要求、配置、工具等 明确测试的系统&#xff1a; 本文档主要指的是web应用。 明确测试要求&#xff1a; 用户提出性能测试&#xff0c;例如&#xff0c;网站首页页面响应时间在3S之内&#xff0c;主要的业务操作时间小于10s&#xff0c;支持300用户在…

Android Studio实现内容丰富的安卓宠物用品商店管理系统

获取源码请点击文章末尾QQ名片联系&#xff0c;源码不免费&#xff0c;尊重创作&#xff0c;尊重劳动。 项目编号128 1.开发环境android stuido jdk1.8 eclipse mysql tomcat 2.功能介绍 安卓端&#xff1a; 1.注册登录 2.系统公告 3.宠物社区&#xff08;可发布宠物帖子&#…

this是什么?为什么要改变this?怎么改变 this 指向?

目录 this 是什么&#xff1f; 箭头函数中的 this 为什么要改变 this 指向&#xff1f; 改变 this 指向的三种方法 call(无数个参数) apply(两个参数) bind(无数个参数) this 是什么&#xff1f; 在对象方法中&#xff0c;this 指的是所有者对象&#xff08;方法的拥有者…

提升口才表达能力的重要性与途径

提升口才表达能力的重要性与途径 口才表达能力&#xff0c;即一个人通过口头语言准确、流畅、生动地传达思想、情感和观点的能力&#xff0c;是现代社会中不可或缺的一项基本技能。无论是在职场沟通、人际交往还是公共场合发言&#xff0c;优秀的口才表达能力都能为我们带来诸…

力扣hot100:416.分割等和子集(组合/动态规划/STL问题)

组合数问题 我们思考一下&#xff0c;如果要把数组分割成两个子集&#xff0c;并且两个子集的元素和相等&#xff0c;是否等价于在数组中寻找若干个数使之和等于所有数的一半&#xff1f;是的&#xff01; 因此我们可以想到&#xff0c;两种方式&#xff1a; ①回溯的方式找到t…

vanna:基于RAG的text2sql框架

文章目录 vanna简介及使用vanna的原理vanna的源码理解总结参考资料 vanna简介及使用 vanna是一个开源的利用了RAG的SQL生成python框架&#xff0c;在2024年3月已经有了5.8k的star数。 Vanna is an MIT-licensed open-source Python RAG (Retrieval-Augmented Generation) fram…

探讨大世界游戏的制作流程及技术——大场景制作技术概况篇

接上文&#xff0c;我们接下来了解一下大世界场景制作技术有哪些&#xff0c;本篇旨在给大家过一遍目前业界的做法&#xff0c;能让大家有一个宏观的知识蓝图。实际上&#xff0c;针对不同的游戏类型和美术风格&#xff0c;制作技术在细节上有着非常大的不同&#xff0c;业界目…

HarmonyOS NEXT应用开发—自定义视图实现Tab效果

介绍 本示例介绍使用Text、List等组件&#xff0c;添加点击事件onclick,动画&#xff0c;animationTo实现自定义Tab效果。 效果预览图 使用说明 点击页签进行切换&#xff0c;选中态页签字体放大加粗&#xff0c;颜色由灰变黑&#xff0c;起到强调作用&#xff0c;同时&…

手撕算法-队列实现栈And栈实现队列

手撕算法-队列实现栈And栈实现队列 两个栈实现队列两个队列实现栈包含min函数的栈 两个栈实现队列 分析&#xff1a;转换数据方向&#xff0c;第一个栈写&#xff0c;第二个栈读。 代码&#xff1a; import java.util.*; import java.util.Stack;public class Solution {Sta…

PyCharm中如何使用不同的虚拟环境

1. 简介 有些项目用老的运行环境&#xff0c;而有些项目用新的运行环境&#xff0c;那么我们在运行这些代码&#xff08;比如跑对比实验的时候&#xff09;如何进行切换呢&#xff0c;这时候就可以使用虚拟环境啦 2. 虚拟环境的创建 首先启动Anaconda Prompt 并在其中执行如…

调皮的String及多种玩法(下部)

&#x1f468;‍&#x1f4bb;作者简介&#xff1a;&#x1f468;&#x1f3fb;‍&#x1f393;告别&#xff0c;今天 &#x1f4d4;高质量专栏 &#xff1a;☕java趣味之旅 欢迎&#x1f64f;点赞&#x1f5e3;️评论&#x1f4e5;收藏&#x1f493;关注 &#x1f496;衷心的希…

LeetCode 面试经典150题 26.删除有序数组中的重复项

题目&#xff1a; 给你一个 非严格递增排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元素的个数。 考虑 nums 的唯一元素的数量…

C# SM2加解密 ——国密SM2算法

SM2 是国家密码管理局组织制定并提出的椭圆曲线密码算法标准。 本文使用第三方密码库 BouncyCastle 实现 SM2 加解密&#xff0c;使用 NuGet 安装即可&#xff0c;包名&#xff1a;Portable.BouncyCastle&#xff0c;目前最新版本为&#xff1a;1.9.0。 using Org.BouncyCastl…

相机sd卡照片删除后数据恢复,相机sd卡中的照片被删除后如何恢复数据

当我们使用相机拍摄照片时&#xff0c;有时会不小心删除了一些重要的照片。这可能是因为误操作、SD卡故障或者其他原因。无论是珍贵的照片、还是重要的工作文件&#xff0c;被删除后&#xff0c;我们往往会感到焦虑和失望。相机sd卡中的照片被删除后如何恢复数据&#xff1f;幸…

Java项目:54 springboot工资信息管理系统453

作者主页&#xff1a;源码空间codegym 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 本系统的使用角色可以被分为用户和管理员&#xff0c; 用户具有注册、查看信息、留言信息等功能&#xff0c; 管理员具有修改用户信息&#…

洛谷P1097 [NOIP2007 提高组] 统计数字

#先看题目 题目描述 某次科研调查时得到了 n 个自然数&#xff0c;每个数均不超过 1.5109。已知不相同的数不超过 个&#xff0c;现在需要统计这些自然数各自出现的次数&#xff0c;并按照自然数从小到大的顺序输出统计结果。 输入格式 共 n1 行。 第一行是整数 n&#x…