Python解题 - CSDN周赛第29期 - 争抢糖豆
创始人
2024-05-26 14:38:14
0

本期问哥是志在必得,这本算法书我已经觊觎许久,而之前两次因为种种原因未能如愿。因此,问哥这几天花了不少时间,把所有之前在每日一练做过的题目重新梳理了一遍。苦心人,天不负,感谢官方大大!


第一题:订班服

小A班级订班服了! 可是小A是个小糊涂鬼,整错了好多人的衣服的大小。 小A只能自己掏钱包来补钱了。 小A想知道自己至少需要买多少件衣服。

输入描述:第一行输入一个整数n。(1<=n<=100)表示衣服的数量。 以下n行输入n个尺码。表示订单中衣服的尺码。 接下来n行输入n个尺码。小A订的衣服尺码。 尺码表:M,S,L,XL,XLL,XLLL,XLLLL,XLLLLL。

输出描述:输出至少需要买多少件衣服。

示例:

示例
输入

2

XL
X
M
X

输出1

分析

简单题。需要 n 件衣服,买了 n 件衣服,所以只要把需要的 n 件衣服每种尺码的个数,减去已购买的衣服相应尺码的个数,如果大于0就说明该尺码还需要购买。所以最直接的办法就是用哈希表记录每种尺码的个数,然后再逐个检查。

Python已经提供了内置Counter类,可以自动生成字典,而且支持加减法,连逐个检查这一步都省去了:直接相减,剩下的数字加在一起就是答案。

参考代码

n = int(input().strip())
arr1 = [input().strip() for _ in range(n)]
arr2 = [input().strip() for _ in range(n)]
from collections import Counter
clothes = Counter(arr1) - Counter(arr2)
print(sum(clothes.values()))

第二题:争抢糖豆

抓糖豆,小Q与小K都喜欢吃糖豆。 但是糖豆分两种,超甜糖豆和普通糖豆。 现在有w个超甜糖豆和b个普通糖豆。 小Q和小K开始吃糖豆,他们决定谁先吃到超甜糖豆谁就获胜。 小K每次吃的时候会捏碎一颗糖豆。 小Q先吃,小Q想知道自己获胜的概率。 如果两个人都吃不到超甜糖豆小K获胜。

输入描述:输入两个整数w,b。(0<=w,b<=1000)

输出描述:答案保留9位小数。

示例:

示例
输入1 3
输出0.500000000

分析

也是以前考过的老题了。可以用递归或动态规划来做,但本题的状态转移不太容易一眼发现,所以可能不少人会觉得难。因为问到概率(胜率),所以本质上还是需要用数学来表达。

以动态规划为例(递归容易超时),我们用 dp[w][b] 表示当有 w 颗超甜糖豆,和 b 颗普通糖豆时自己的胜率。因为先吃到超甜糖豆就获胜了,所以自己要想获胜,只能分成两种情况:

  1. 先吃到超甜糖豆,概率是 \frac{w}{w+b}  ,此情况下直接获胜;
  2. 先吃到普通糖豆,但是对手也吃到普通糖豆,所以游戏继续,自己还有获胜的可能。(这里有一个特判的情况:如果只有一颗普通糖豆,而自己先吃到普通糖豆的话,无论如何也是输,后面自然就不用算了。)因此,自己和对手都吃到普通糖豆的概率是 \frac{b}{w+b} * \frac{b-1}{w+b-1}  。(如果一时看不懂可以多琢磨几遍,乘号左边是自己吃到普通糖豆的概率,右边是自己吃完后对方也吃到普通糖豆的概率,看懂了再继续。)

如果没有“捏碎糖豆”的操作,分析到这就结束了,状态转移就是把这两种情况的胜率加在一起,方程如下:

dp[w][b]=\frac{w}{w+b}+\frac{b}{w+b}*\frac{b-1}{w+b-1}*dp[w][b-2]

(因为自己和对手总共吃了两颗普通糖豆,所以上面第二种情况的概率还要乘以 dp[w][b-2] 才是胜率。)

如果上面的内容理解了,我们再来分析“捏碎糖豆”的情况。

捏碎糖豆也有两种情况:

  1. 捏碎了普通糖豆。影响不大,但是上面的第二种状态要接着乘上捏碎普通糖豆的概率,再乘以 dp[w][b-3] 。合在一起的胜率就是 \frac{b}{w+b}*\frac{b-1}{w+b-1}*\frac{b-2}{w+b-2}*dp[w][b-3] 。
  2. 捏碎了超甜糖豆。则二人虽不分胜负,理论上自己还存在胜利的可能(如果还剩下超甜糖豆的话),但是同样地,状态转移方程变了,胜率变成了:\frac{b}{w+b}*\frac{b-1}{w+b-1}*\frac{w}{w+b-2}*dp[w-1][b-2] 。

这两种捏碎糖豆的情况属于同一决策层级,可以加在一起。于是把捏碎糖豆考虑进来,得到最终的状态转移方程如下:

dp[w][b]=\frac{w}{w+b}+\frac{b}{w+b}*\frac{b-1}{w+b-1}*(\frac{b-2}{w+b-2}*dp[w][b-3]+\frac{w}{w+b-2}*dp[w-1][b-2])

此外,如之前所述,还要考虑几个特判的情况:

  1. 没有超甜糖豆,胜率为0,不用计算。
  2. 没有普通糖豆,胜率100%。
  3. 只有一颗普通糖豆,胜率为\frac{w}{w+b} ,因为如果自己先吃到普通糖豆,必输。

很显然,上面第二、三可以合并,而第一条可以在初始化的时候把 dp[0][j],0\leq j\leq b 的时候设置为0。代码如下:

参考代码

w, b = map(int, input().split())
dp = [[0]*(b+1) for _ in range(w+1)]
for i in range(1, w+1):for j in range(b+1):if j <= 1:dp[i][j]=i/(i+j)else:dp[i][j]=i/(i+j)+j/(i+j)*(j-1)/(i+j-1)*((j-2)/(i+j-2)*dp[i][j-3]+i/(i+j-2)*dp[i-1][j-2])
print(f"{dp[w][b]:.9f}")

输出结果的时候要注意,题目要求必须保留9位小数,空位用0补全,所以要设置占位符。


第三题:走楼梯

现在有一截楼梯,根据你的腿长,你一次能走 1 级或 2 级楼梯,已知你要走 n 级楼梯才能走到你的目的楼层,请实现一个方法,计算你走到目的楼层的方案数。

输入描述:输入整数n。(1<=n<=50)

输出描述:输出方案数。

示例:

示例
输入5
输出

8

分析

很明显,答案是斐波那契数列。因为一次只能走 1 级或 2 级楼梯,所以第 n 级阶梯可以由 n-1 级阶梯走过来,也可以由 n-2 级阶梯走过来。如果用函数 f(n) 表示走上第 n 级阶梯的方案数,可得

f(n) = f(n-1)+f(n-2)

因为本题 n 范围较小(1<=n<=50),所以使用递归来做应该也没问题。不过更常用的做法是递推,也算是斐波那契的模板题了。

参考代码

n = int(input().strip())
a = b = 1
for _ in range(n):a, b = b, a+b
print(a)

第四题:打家劫舍

一个小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。

输入描述:输入一个正整数n代表房屋的数量(n≤100),接着输入n个非负整数代表每间房屋的现金数量

输出描述:小偷能偷取的最大金额。

示例:

示例
输入4
1 2 3 1
输出4

分析

力扣原题,经典的打家劫舍系列,但凡刷过点题的相信都做过。而且这里选取的是该系列最简单的一道,动态规划入门题,相关题解太多了,这里问哥只简单说两句吧。

因为相邻的房屋不能同时被盗,所以小偷在当前房屋只有两种选择:不偷当前房屋——继承上个房屋可偷取的的最大金额,偷当前房屋——上上个房屋可偷取的最大金额(因为上个房屋不能偷)加上当前房屋的金额,而当前房屋可偷取的最大金额就等于这两种选择中较大的金额。

如果用 f(i) 代表当前房屋可偷取的最大金额,H_{i} 表示当前房屋的金额,可用公式表示如下:

f(i)=max({f(i-1),f(i-2)+H_{i}})

类似斐波那契数列,可以看出 f(i) 仅由 f(i-1) 和 f(i-2) 得到(H_{i} 是给定数组),所以可以使用滚动数组优化空间,换句话说就是只需要额外两个变量循环保存 f(i-1) 和 f(i-2) 的值即可。

参考代码

n = int(input().strip())
arr = [int(item) for item in input().strip().split()]
a = b = 0
for i in arr:a, b = b, max(b, a+i)
print(b)

相关内容

热门资讯

emui 安卓系统对应关系,E... 你有没有发现,每次打开你的华为手机,那个界面看起来是不是特别顺眼?那是因为华为的EMUI系统,它就像...
永诺安卓系统相机,功能解析与使... 你有没有发现,手机拍照已经成为我们生活中不可或缺的一部分?而在这其中,永诺安卓系统的相机功能可是相当...
tinder安卓版系统错误,揭... 最近在使用Tinder安卓版的时候,你是不是也遇到了一些让人头疼的系统错误呢?别急,今天就来和你聊聊...
htc安卓系统怎么更新系统,轻... 亲爱的HTC安卓用户们,你是不是也和我一样,时不时地想给手机来个“大变身”,让它焕然一新呢?没错,今...
安卓最新发布系统,颠覆性更新与... 你知道吗?最近安卓系统又来了一次大变身,这可是科技圈里的大事哦!安卓最新发布的系统,简直就像是一个全...
华为不升级安卓系统,开启自主操... 你知道吗?最近有个大新闻在科技圈里炸开了锅,那就是华为决定不再升级安卓系统!这可不是一个小决定,它背...
安卓保护系统停止运行,紧急排查... 亲爱的手机用户们,你们有没有遇到过这样的情况:手机突然间变得不正常了,安卓保护系统竟然停止运行了?这...
安卓系统记录仪,智能行车安全守... 你有没有想过,开车的时候,那些瞬间发生的事情,就像电影里的精彩片段,一闪而过,却让人回味无穷?别急,...
安卓13系统怎样升级,全面解析... 你有没有发现,你的安卓手机最近是不是有点儿“蔫儿”了?别急,别急,我来告诉你怎么给它来个“大变身”—...
安卓手机进去系统花屏,安卓手机... 手机屏幕突然花屏了,是不是瞬间感觉整个世界都变得不美好了呢?别急,今天就来和你聊聊安卓手机进入系统时...
安卓手机 系统怎么更新,体验最... 亲爱的手机控们,你是不是也和我一样,时不时地想给安卓手机来个“美容”大变身呢?没错,说的就是系统更新...
妈妈手机推荐安卓系统,安卓系统... 亲爱的妈妈们,是不是在为给家里的宝贝挑选一款合适的手机而烦恼呢?别急,今天我就来给你详细介绍一下几款...
oppo安卓版系统设置,全面解... 亲爱的手机控们,你是不是也和我一样,对OPPO安卓版系统的设置充满了好奇?想要让你的OPPO手机更加...
安卓系统是什么cp,CP架构下... 你有没有想过,你的手机里那个默默无闻的安卓系统,其实就像是一个超级贴心的CP(情侣搭档)呢?没错,就...
系统垃圾清理大师 安卓,安卓手... 手机里的垃圾文件是不是让你头疼不已?别急,今天我要给你介绍一位安卓系统里的“清洁小能手”——系统垃圾...
安卓系统分为几层,安卓系统分层... 你知道吗?安卓系统,这个陪伴我们手机生活的“小助手”,其实它内部结构可是相当复杂的呢!今天,就让我带...
系统最像苹果的安卓,揭秘最像苹... 你有没有发现,现在的安卓手机越来越像苹果了?没错,就是那个以简洁设计和流畅体验著称的苹果。今天,就让...
安卓更新13系统游戏,性能升级... 你知道吗?最近安卓系统又来了一次大变身,那就是安卓13系统!这次更新可是带来了不少惊喜,尤其是对那些...
安卓系统开机出错了,安卓系统开... 手机突然开不了机了,这可怎么办?别急,让我来帮你分析一下安卓系统开机出错的那些事儿。一、安卓系统开机...
vovg是安卓系统吗,安卓系统... 你有没有听说过Vovg这个操作系统?最近,这个名词在数码圈里可是引起了不小的热议呢!很多人都在问,V...