【蓝桥杯国赛真题笔记】Python(2)

简介: 【蓝桥杯国赛真题笔记】Python

3.包子凑数


image.png


问题描述:


小明几乎每天早晨都会在一家包子铺吃早餐。他发现这家包子铺有N种蒸笼,其中第i种蒸笼恰好能放Ai个包子。每种蒸笼都有非常多笼,可以认为是无限笼。


 每当有顾客想买X个包子,卖包子的大叔就会迅速选出若干笼包子来,使得这若干笼中恰好一共有X个包子。比如一共有3种蒸笼,分别能放3、4和5个包子。当顾客想买11个包子时,大叔就会选2笼3个的再加1笼5个的(也可能选出1笼3个的再加2笼4个的)。


 当然有时包子大叔无论如何也凑不出顾客想买的数量。比如一共有3种蒸笼,分别能放4、5和6个包子。而顾客想买7个包子时,大叔就凑不出来了。


 小明想知道一共有多少种数目是包子大叔凑不出来的。

image.png



问题分析: 结合这道题的结论:


两个数p,q互质,那么p和q最大不能表示的数字为pq-(p+q)


我们大胆推论,如果给出的n种包子数目不互质,那么就有inf(无穷)多个不能表示


否则 即gcd(a,b,c,d..)=1,就有有限个数字不能表示


包子数目最大是100,那么与100互质的最大是99。


在这n种包子数目中,总能取出两个最大的,即为p,q,那么pq-(p+q)+1开始以后的数目一定都可以表示,所以我们只需要统计[1,pq-(p+q])之内有多少个不能表示的数字


不妨令p>q,那么pq-(p+q)<pq<=9900,所以无论给出多少种互质包子数目,我们至多需要在[1,9900]这个范围内去搜(这里需要好好理解一下,想象一下区间)


下面讨论如何去搜:由于有一种包子无限个,想到完全背包问题


二维完全背包和二维01背包的区别在于 动态规划-完全背包问题_WUST_XIAO的博客-CSDN博客

image.png


可以这样理解:01背包问题中dp[i][j]设为容量为i前j个物品的最大价值


完全背包问题dp[i][j]代表容量为i前j种物品的最大价值


所以在01背包dp[i][j]在递推关系中,可能转移到dp[i-weight[j]][j-1]+val[j],j减掉了1,因为物品有限;


而在完全背包问题中,可能转移到dp[i-weight[j]][j]+val[j],j不变,因为物品无限,理解为预留weight[j]的空间,用前j种物品能够凑出来的最大值,那么这样理解,第j个物品就不止用一次。然后本题就可以迎刃而解~


AC代码


n=int(input())
a=[0]*n
for i in range(n):
    a[i]=int(input())
def gcd(a,b):
    while b:
        a,b=b,a%b
    return a
def l_gcd(nums):
    if len(nums)==1:
        return nums[0]
    elif len(nums)==2:
        return gcd(nums[0],nums[1])
    else:
        return gcd(l_gcd(nums[:len(nums)//2]),l_gcd(nums[len(nums)//2:]))
if l_gcd(a)!=1:#这n个数字的最大公约数不等于1
    print('INF')
else:#这n个数字的最大公约数等于1,有解
    #区间确定,取n个数字里面最大的两个p,q
    #那么在正整数集合,pq-(p+q)+1开始之后均可以表示,所以[1,9900]
    ans,res=1,0
    a.insert(0,0)
    top=10000
    dp=[[0]*(n+1)for i in range(0,top+1)]#目标为i,前j种包子(每种包子数量无限)能不能凑出来
    for i in a:
        dp[i]=[1]*(n+1)
    for i in range(1,top+1):
        for j in range(1,n+1):
            if a[j]>i:
                dp[i][j]=dp[i][j-1]
            else:
                dp[i][j]=max(dp[i][j-1],dp[i-a[j]][j])
    v=0
    for k in range(1,len(dp)):
        if 1 not in  dp[k]:
            v+=1
    print(v)

         


4.K倍区间(省赛题)


问题描述:给定一个长度为N的数列,A1, A2, ... AN,如果其中一段连续的子序列Ai, Ai+1, ... Aj(i <= j)之和是K的倍数,我们就称这个区间[i, j]是K倍区间。


你能求出数列中总共有多少个K倍区间吗?


image.png


问题分析:已知(a+b)%p=(a%p+b%p)%p


推导过程,设a=xp+y,b=mp+n,那么(a+b)%p=((x+m)p+y+n)%p=(y+n)%p


即(a%p+b%p)%p 得证


那么,对于本题而言,还需要用到前缀和 不妨令Si>Sj,那么Si-Sj代表的和为区间[j+1,i]


按照提议(Si-Sj)%k==0 那么(Si%k-Sj%k)%k==0


那么Si%k=Sj%k,所以利用这个条件:


根据样例输入,我们存入数组a=[1,2,3,4,5]


前缀和数组[1,3,6,10,15]  ,前缀和取余数组z=[1,1,0,0,1]


我们发现z[0],z[1],z[4]=1,那么[1],[1,4],[2,4]这三个区间都符合,懂排列组合的应该明白了等于C(n,2),到这里有三种


然后我们发现z[2],z[3]=0 那么 [3]这个区间符合,到这里加上前面3种一共4种


还有两种去哪了?把区间都列一下,会发现,正是由于我们的下标从0开始,那么区间的左端点必定大于等于1,那么就无法取到下标0,所以我们在数组a最前面插入一个0,前缀和数组和前缀和取余数组相应变化。会发现,现在0正好有3个,C(3,2)=3


加起来一共6种 符合预期,所以利用好样例,根据自己程序的实际和真正的要求做对比,发现缺了什么,想一想,大胆补上去条件,问题往往都可以迎刃而解。



AC代码


n,k=map(int,input().split())
a=[0]*n
for i in range(n):
    a[i]=int(input())
a.insert(0,0)
s=[0]*(n+1)#前缀和
s[0]=a[0]
for i in range(1,n+1):
    s[i]=s[i-1]+a[i]
#对s[i] mod k,代表前i项和对k取余后的余数
z=[0]*(n+1)
for i in range(n+1):
    z[i]=s[i]%k
ans=0
p=dict()
for i in range(n+1):
    if z[i] not in p.keys():
        p[z[i]]=1
    else:
        p[z[i]]+=1
for j in p.values():
    ans+=int(j*(j-1)/2)
print(ans)


5.青蛙跳杯子

image.png



奈何小郑没有写出来,是一个BFS模板题,求最少次数。66分超时了


但还是总结一下吧:BFS需要队列queue,pre记录前驱结点,searched记录访问过的结点,judge函数判断是否合法,关键点就这么些。(代码可以看看,没AC)


st=input()
end=input()
queue=[st]
step=0
searched=[]
pre={}
dx=[-3,-2,-1,1,2,3]
def judge(new):
    global searched,pre
    if new not in searched and new not in pre.keys():
        return True
    return False
def change(tmp,now_index,new_index):
    y=list(tmp)
    y[now_index],y[new_index]=y[new_index],y[now_index]
    return ''.join(y)
while queue:
    tmp=queue.pop(0)
    if tmp==end:
        step=1
        while pre[tmp]!=st:
            step+=1
            tmp=pre[tmp]
        print(step)
        break
    else:
        #BFS搜6个方向,索引*位置
        now_index=tmp.index('*')
        for i in range(6):
            new_index=now_index+dx[i]
            if 0<=new_index<=len(tmp)-1:
                new=change(tmp,now_index,new_index)
                if judge(new):
                    searched.append(new)
                    queue.append(new)
                    pre[new]=tmp


学习犹如逆水行舟,不进则退,希望大家都能在这次省赛中拿到理想成绩。


有任何不懂欢迎留言提问 !


目录
相关文章
|
1月前
|
Python
【python】】Python 的 queue 模块使用笔记
【python】】Python 的 queue 模块使用笔记
24 0
|
1月前
|
Python
Python笔记9 类
本文是作者的Python复习笔记第九篇,深入探讨了Python中的类和面向对象编程。文中详细解释了如何创建类、实例化对象、定义和使用类方法,以及类的继承、重写方法和嵌套类的使用。此外,还讨论了类模块的导入和导出,包括处理类之间的依赖关系。通过示例代码,文章展示了类在Python编程中的应用和重要性。
23 0
|
1月前
|
存储 Python
Python笔记8 函数
本文是作者的Python复习笔记第八篇,全面介绍了Python中的函数定义与使用,包括函数的参数传递(位置参数、关键字参数、默认参数、列表参数、任意数量参数和关键字参数)、函数的返回值以及如何创建和调用函数库(模块),并提供了丰富的示例代码。
23 0
|
1月前
|
Python
Python笔记7 输入与输出
本文是作者的Python复习笔记第七篇,主要介绍了Python中的输入与输出操作。文中详细解释了如何使用input()函数进行用户输入,包括添加多行字符串提示和字符串转列表输入的方法,以及如何使用print()函数进行格式化输出,提供了多种格式化字符串的示例。
25 0
|
1月前
|
存储 Python
Python笔记6 字典
本文是作者的Python复习笔记第六篇,专注于Python中的字典(dictionary)数据结构。文中详细解释了字典的创建和基本操作,包括访问、修改、添加和删除键值对的方法。此外,还介绍了如何遍历字典的键值对、键或值,并探讨了字典的高级用法,如字典列表、在字典中存储列表以及字典的嵌套使用。文中通过示例代码演示了字典在实际编程中的应用,帮助读者更好地理解和掌握字典这一重要的数据结构。
36 0
|
1月前
|
Python
Python笔记5 条件判断
本文是作者的Python复习笔记第五篇,主要介绍了Python中的条件判断语句。文中详细解释了if、if-else以及if-elif-else结构的用法,包括如何使用等于(==)和不等于(!=)操作符进行条件判断,如何通过and和or进行多条件判断,以及如何使用in和not in关键字检查列表中是否存在特定值。此外,文中还强调了在某些情况下省略else部分可以避免执行不合适的数据导致的命令执行,使代码更加清晰。
22 0
|
1月前
|
Python
Python笔记4 循环
本文是作者的Python复习笔记第四篇,专注于Python中的循环概念。文中详细解释了for循环和while循环的使用方法,包括如何通过循环遍历列表、使用range()函数和list()函数创建列表、列表解析法、while循环的基本使用、使用break和continue语句控制循环流程,以及如何为循环设置状态标志。此外,还提供了多个示例代码来演示循环在实际编程中的应用。
23 0
|
移动开发 Shell Python
【蓝桥杯国赛真题笔记】Python(1)
【蓝桥杯国赛真题笔记】Python
154 0
【蓝桥杯国赛真题笔记】Python(1)
|
5天前
|
数据采集 机器学习/深度学习 人工智能
Python编程入门:从零基础到实战应用
【9月更文挑战第15天】本文将引导读者从零开始学习Python编程,通过简单易懂的语言和实例,帮助初学者掌握Python的基本语法和常用库,最终实现一个简单的实战项目。文章结构清晰,分为基础知识、进阶技巧和实战应用三个部分,逐步深入,让读者在学习过程中不断积累经验,提高编程能力。
|
1天前
|
数据可视化 Python
Python编程中的数据可视化技术
【9月更文挑战第19天】在数据驱动的时代,将复杂的数据集转化为直观易懂的视觉表达至关重要。本文将深入探索Python中的数据可视化库,如Matplotlib和Seaborn,并指导读者如何运用这些工具来揭示数据背后的模式和趋势。文章不仅会介绍基础图表的绘制方法,还将讨论高级技巧以提升图表的信息丰富度和吸引力。