【Python练习题 020】 求1+2!+3!+...+20!的和

--------------------------------------------------

据说这题是“累积累加”的问题,把“1+2!+3!+...+20!”展开就变成:

1

1*2

1*2*3

1*2*3*4

……

1*2*3*4……*20

弄懂了这规律,问题就好解决了。代码如下:

sum = 0
x = 1
for i in range(1,21):
x = x * i
sum = sum + x
print(sum)

输出结果如下:

2561327494111820313

【2016-10-18 更新】------------------------------------------------

感谢 rm-rf 的指教,用递归、lambda、functools.reduce() 这三把斧头为我们提供了新思路。不过,新知识太多了,一时我接受不了,还是一样一样来吧。

先不管递归,我们来看看使用了 lambda 和 functools.reduce() 的代码吧:

import functools

sum = 0
for i in range(1,21):
sum = sum + functools.reduce(lambda x,y: x*y, range(1, i+1))
print(sum)

根据网上的资料,上述代码中,lambda x,y: x*y 的意思是:有一个函数,接受x, y两个参数做乘法。也就是说,把函数压缩到1行,真是很简洁。

而 functools.reduce() 也很神奇,大概结构是 functools.reduce(函数, 序列, 初始参数),其中括号里的函数需要2个参数,如果没有指定初始参数,就先从序列中拿出2个传递给函数,之后把结果作为第1个参数,再从序列中拿出第3个项目作为第2个参数一起传递给函数,如此循环至序列空空为止。如果指定初始参数,则这个初始参数作为第1个参数,然后从序列中拿出1个作为第2个参数,传递给函数。请参见官方文档

上述代码中,functools.reduce(lambda x,y: x*y, range(1,i+1)) 意思就是说:先从 range(1,i+1) 拿出2个元素,作为参数传递给乘法函数 lambda x,y: x*y,计算出的结果摇身变成参数,再从 range(1, i+1) 拿出1个元素,两个配一起传递给函数重新计算。假如是 range(1, 4),这个函数就会计算 (1*2)*3。

有了以上做铺垫,外面再套个 for 从1循环到20,就能得出题解。

弄明白 lambda 和 functools.reduce() 之后,就可以用递归来解决这道题了。

所谓“递归”,就是在函数体中调用自身。也就是说,在函数的计算式里,有一个元素是需要另外拿出来,放进这个函数里进行计算的,而在计算这个元素时,可能又有一个元素还得拿出来单独再计算,直到所有元素都不符合条件,循环也就结束了。

就像本题,倒着写就成了 20! + 19! + 18! + ... + 3! + 2! + 1。如果每个阶乘 n! 都写成 f(n),计算 f(n) 时需要先知道 f(n-1) 的结果,而计算 f(n-1) 时同样又需要先计算 f(n-2) 的值,如此循环,直至 n == 1,递归结束 …… 我觉得我根本讲不清楚!!不管了,上代码:

import functools

def f(n):
if n == 1:
return 1
else:
return functools.reduce(lambda x,y: x*y, range(1, n+1)) + f(n-1)
print(f(20))

唉,其实还是蛮难理解的。多做点练习,也许能逐渐掌握吧……

++++++++++++++++++++++++++++++++++++++

题目出处:编程语言入门经典100例【Python版】

Python练习题 020:累积累加的更多相关文章

  1. Python练习题 021:递归方法求阶乘

    [Python练习题 021] 利用递归方法求5!. ---------------------------------------------- 首先得弄清楚:5! 指的是"5的阶乘&qu ...

  2. Python练习题 004:判断某日期是该年的第几天

    [Python练习题 004]输入某年某月某日,判断这一天是这一年的第几天? ---------------------------------------------- 这题竟然写了 28 行代码! ...

  3. Python练习题-1.使用匿名函数对1~1000求和,代码力求简洁。

    Python 练习 标签(空格分隔): Python Python练习题 Python知识点 一.使用匿名函数对1~1000求和,代码力求简洁. 答案: In [1]: from functools ...

  4. Python练习题 002:奖金计算

    [Python练习题 002]企业发放的奖金根据利润提成.利润(I)低于或等于10万元时,奖金可提10%:利润高于10万元,低于20万元时,低于10万元的部分按10%提成,高于10万元的部分,可可提成 ...

  5. Python练习题 028:求3*3矩阵对角线数字之和

    [Python练习题 028] 求一个3*3矩阵对角线元素之和 ----------------------------------------------------- 这题解倒是解出来了,但总觉得 ...

  6. Python练习题 027:对10个数字进行排序

    [Python练习题 027] 对10个数字进行排序 --------------------------------------------- 这题没什么好说的,用 str.split(' ') 获 ...

  7. Python练习题 026:求100以内的素数

    [Python练习题 026] 求100以内的素数. ------------------------------------------------- 奇怪,求解素数的题,之前不是做过了吗?难道是想 ...

  8. Python练习题 025:判断回文数

    [Python练习题 025] 一个5位数,判断它是不是回文数.即12321是回文数,个位与万位相同,十位与千位相同. ---------------------------------------- ...

  9. Python练习题 024:求位数及逆序打印

    [Python练习题 024] 给一个不多于5位的正整数,要求:一.求它是几位数,二.逆序打印出各位数字. ---------------------------------------------- ...

随机推荐

  1. [QZOI2019]Game 题解

    QZOI2019 CSP-S模拟赛 T1 错误的贪心导致考场上只有10pts... 看来以后贪心还是需要先证明啊 题目描述 小A和小B在玩一个游戏,他们两个人每人有 $n$ 张牌,每张牌有一个点数,并 ...

  2. rake

    ruby-rake https://rubygems.org/gems/rake 官方文档 https://ruby.github.io/rake/ http://docs.seattlerb.org ...

  3. 题解 洛谷P2434 【[SDOI2005]区间】

    本题的贪心策略是以区间起点位置由小到大排序,然后开始合并. 区间按起点顺序由小到大排序,可以最大化合并成功的可能. 这个脑补应该不难想出来.(读者自证不难 直接上代码: #include <bi ...

  4. Ajax提交数据判断员工编号是否存在,及自动填充与员工编号所对应的员工姓名。

    JSP页面中所需要的JavaScript事件及Ajax <script type="text/javascript"> function checkEmpNo(id){ ...

  5. SpringMVC-08-整合SSM之基本环境搭建

    8. 整合SSM 环境要求 IDEA MySQL 5.5 Tomcat 9 Maven 3.5.2 要求: 需要熟练掌握MySQL数据库,Spring,JavaWeb及Mybatis知识,简单的前端知 ...

  6. 剑指 Offer 46. 把数字翻译成字符串

    题目描述 给定一个数字,我们按照如下规则把它翻译为字符串:0 翻译成 "a" ,1 翻译成 "b",--,11 翻译成 "l",--,25 ...

  7. 字符串split的用法

    拆分字符串:张三:20|李四:40|王五:40 这个可以使用两次分割,第一次使用 | 分割,放到arr数组里,然后使用循环对arr[i]进行使用:分割 public static void main( ...

  8. 轻松上手SpringBoot Security + JWT Hello World示例

    前言 在本教程中,我们将开发一个Spring Boot应用程序,该应用程序使用JWT身份验证来保护公开的REST API.在此示例中,我们将使用硬编码的用户和密码进行用户身份验证. 在下一个教程中,我 ...

  9. Mybatis的Mapper中的方法为什么不能重载?

    目录 前言 环境配置 错误示范 为什么不能重载? 如何找到XML中对应的SQL? 总结 前言 在初入门Mybatis的时候可能都犯过一个错误,那就是在写Mapper接口的时候都重载过其中的方法,但是运 ...

  10. [Java并发编程之美]第2章 并发编程的其他基础知识 补充知识

    基本概念 并行与并发 并行:单位时间内多个任务同时执行(多核CPU). 并发:同一时间段内多个任务同时都在执行(CPU时间片轮转). 线程安全性问题 线程安全问题:指多个线程同时读写一个共享资源而没有 ...