蓝桥杯 省赛 杨辉三角形 python组 找规律+二分查找+组合数-爱代码爱编程
2.渐增性
越往中间数字越大,除最外层1外,越往下数字越大。
3.组合数
杨辉三角形里面的每一个元素都能用组合数表示。第N行的第M列可以表示成C(N-1,M-1),如6在第5行的第3列,它对应的组合数就是C(5-1,3-1),即C(4,2)。
=================================================================
因为要找出第一次N出现的位置,根据对称性可知,N出现的位置必定在左边,因此只考虑左半边位置即可。因为越靠中间的数越大,所以我们可以从最中间的数,也就是从对称轴位置的数开始找。怎么找呢?斜着找。没错,就是斜着找。
我们先将三角形的右半部分去掉,然后再区分开每一斜行。
为什么要斜着找?
因为越靠近里面的斜行每一元素的值越大,而且总是较先出现的。以上图为例,6在倒数第二斜行出现了,他对应的位置是第5行第3列,也在倒数第三斜行出现了,对的位置是第7行第2列。很显然,倒数第二斜行的6的位置明显后于前者。出现这种情况的原因是每一斜行的增长速度不同,越靠近内行的数值增长速度越快。打个比方,同样做完一道题,内行的15分钟就做完了,外行的可能要花上1个小时。
为什么可以斜着找?
因为它们是有规律的:每一斜行的元素对应的列位置都没变。还是以6为例,6在第3列,6的斜行下一个元素10同样在第三列位置。(6:C(4, 2), 10:C(5, 2))。最后当我们找到元素后根据组合数规律就能反推出该元素在整个杨辉三角形的位置。
解决完上面的疑惑后,就要思考该怎么斜着找的问题了。在思考之前,我们先来看下斜行有哪些性质。
-
在每一斜行中,越往下的元素越大。也就是说在中心轴位置的元素反而是斜行中最小的。(这里作为斜行的起始点)
-
在中心对称轴位置的元素组合数下限是上限的两倍(除1外)。如 2 = C(1,2),6 = C(2,4),10 = C(3:6)…即C(k,2k)
-
斜行同样可以用组合数表示。从全是1的最外层元素开始数(假设是第0斜行),则第 i 斜行的元素可以用组合数C(i, P)表示(P >= 2i,因为斜行的第一个元素就是C(i,2i),见性质2。因此,斜行中每往下一个元素P就加1,i不变)。
怎么斜着找?
因为在内斜行中元素总是较先出现的,所以我们要从内斜行开始从内往外开始找,找到第0行全是1的最外层为止。内到什么程度才行?内到第16行。我们知道在中心对称轴的元素是每一斜行中最小的,它的特点是 C( k, 2k ) 。
如果该斜行最小元素都已经超出10的9次方那么剩下的元素都是大于10的9次方的,也就是说这一斜行是没有意义的,不用考虑。经计算,只有16斜行以内的数才符合条件。
我们已经确定了起始元素,根据杨辉三角的渐增性,越往下元素值越大,说明就是有序的,可以使用二分查找提高查找效率。这里有二分查找和排序的模板,大家可以参考下我的这篇文章。
确定了查找的起点位置后就要确定查找的终点位置。我们以目标值作为我们的组合数下限。回到分析中的第三小点:组合数下限表示元素所在横行数-1,那么如果以目标值作为终点位置的组合数下限已经是非常大了,就算找不到也有第1斜行(全为1的是第0斜行)的公差为1的等差数列守着,所以一定能找到。
因为相同斜行的组合数上限不变,我们不断更换组合数下限的值,直到最后找到目标值即可。找到目标值后,根据此时的组合数上下限,结合杨辉三角组合数性质即可求出元素所在位置。以20: C( 6, 3 )为例,它是,第7行第4个元素。前面6行是个公差为1的等差数列,根据求和公式即可求出6行共有几个元素,最后再加4即为20所在位置。
精度问题: 因为最后输出的是整数,所以最后要使用int将计算结果中小数点后面的数去除。假设一个元素在几十万横行后面找到,那么求他的位置时它的前N项和是非常大的,但他所在的列数可能非常小,如果将他们相加后再转化为整型的话会造成数据丢失,导致与实际结果不符。这样放在蓝桥杯上的话只能够拿八十分。辛辛苦苦做出来的题却因为精度问题不能拿满分,属实可惜。这个问题我也想了好久没找到问题根源,最后看到一篇文章点醒了我,感谢 @Py小郑 。
=================================================================
求组合数
def C(a, b): # a为上限, b为下限
res = 1
for i in range(a):
res *= b / a
当结果大于目标值时无需继续运算,提高效率
if res > target:
return res
b -= 1
a -= 1
return res
二分查找目标元素
def search(k):
起始下限,也就是对称轴位置的元素
low = 2 * k
终点下限
high = target
可能出现high 小于 low 的情况,比如目标值很小,但行数还在十多行的时候。
这时候直接判断该斜行第一个元素也就是对称轴位置的元素的值是否是目标值即可。
文末有福利领取哦~
👉一、Python所有方向的学习路线
Python所有方向的技术点做的整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照上面的知识点去找对应的学习资源,保证自己学得较为全面。
👉二、Python必备开发工具
👉三、Python视频合集
观看零基础学习视频,看视频学习是最快捷也是最有效果的方式,跟着视频中老师的思路,从基础到深入,还是很容易入门的。
👉 四、实战案例
光学理论是没用的,要学会跟着一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。(文末领读者福利)
👉五、Python练习题
检查学习结果。
👉六、面试资料
我们学习Python必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有阿里大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。
👉因篇幅有限,仅展示部分资料,这份完整版的Python全套学习资料已经上传
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!