Python:用filter函数求素数 (再理解generator)
目的:更熟悉应用generator。
参考:https://www.liaoxuefeng.com/wiki/1016959663602400/1017404530360000
素数定义:
素数:质数又称素数。一个大于1的自然数,除了1和它自身外,不能被其他自然数整除的数叫做质数。
方法:
首先,列出从2开始的所有自然数,构造一个序列:
2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, ...
第二步,取序列的第一个数2,它一定是素数,然后用2把序列的2的倍数筛掉:
3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, ...
⚠️此时的序列是从3开始的奇数集合。
第三步,取新序列的第一个数3,它一定是素数,然后用3把序列的3的倍数筛掉:
5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, ...
第四步,取新序列的第一个数5,然后用5把序列的5的倍数筛掉:
7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, ...
后续,不断筛下去,就可以得到所有的素数。
代码:
2是素数,直接从一个奇数序列开始筛选:
# 生成一个奇数序列。从3开始
def _odd_iter():
n = 1
while True:
n += 2
yield n
筛选函数:
- 由上面的分析可知,每次筛选,都除以序列的第一数。
- 条件是x % n > 0。x为序列中被判断的数字,n为该序列的第一个数字。整除结果==0的都不是素树。要从序列中删除。
- 所以,判断表达式是不固定的。定义一个函数,传入参数n, 返回一个lambda表达式用于判断。
# 传入参数n,生成不同的lambda表达式。
def _not_divisible(n):
# 返回一个lambda表达式。用于判断x,是不是素数。
# 素数,只能被1和自身整除的自然数叫做素数。
# 整除就是余数为0。
# 如果被非自身数字整除,即余数为0,那自然不是素数,删除掉。
return lambda x: x % n > 0
主函数:
- 第一行代码:yield 2,返回素数2。
- 第二行代码:创建生成器,第一次运行,它产生奇数序列,这个序列会逐步被筛选,去掉非素数字。
- 循环代码:
- n = next(it), 得到序列的第一个数字。
- yield n, 它是素数。
- 用filter()来筛选, it生成器中的数字。 因为it是可迭代的。并把新的序列赋值给it变量。
⚠️重新理解3。filter函数对生成器对象it进行筛选,其实就是代码组合,把_not_divisible(n)返回的lambda表达式和it自身的代码组合,形成新的对象<filter object>。
⚠️我的理解是,原先的_odd_iter()产生的算法不再用,而是使用增加了lambda条件判断的新的算法,<filter object>。
⚠️这一步并不是真的筛选掉不符合的数字,_ood_iter()只是代表了生成序列数字的算法。只不过这个算法被新的算法替代了。
def primes():
yield 2 #2是素数。
it = _odd_iter() #创建一个<generator object _odd_iter>,代表奇数序列。
while True:
n = next(it) #返回奇数序列的一个数。
yield n #这个数是素数。所以返回。
# 创建一个<filter object>, 代表新序列。
it = filter(_not_divisible(n), it)
由于primes()所代表的算法生成的是一个无限序列,没有退出机制,所以调用时需要设置一个退出循环的条件:
for n in primes():
if n < 1000:
print(n)
else:
break
⚠️primes本身是生成器函数, primes()就是一个生成器,它自身不储存数据,只储存生成数据的算法。
总结;
由本例可知,generator其实就是算法,它不会一次性产生全部数据,而是根据调用的次数,逐步向内存写入数据。
本例把一个生成器对象作为参数传入了filter()函数,其实就是对这个生成器的代码进行更新,即改进算法。最后返回一个新的迭代器。本例返回的是<filter object>。
Python:用filter函数求素数 (再理解generator)的更多相关文章
- python基础——filter函数
python基础——filter函数 Python内建的filter()函数用于过滤序列. 和map()类似,filter()也接收一个函数和一个序列.和map()不同的是,filter()把传入的函 ...
- python中filter函数
python中filter()函数 filter()函数是 Python 内置的另一个有用的高阶函数,filter()函数接收一个函数 f 和一个list,这个函数 f 的作用是对每个元素进行判断 ...
- python的filter()函数
filter()函数是 Python 内置的另一个有用的高阶函数. filter()函数接收一个函数 f 和一个list,这个函数 f 的作用是对每个元素进行判断,返回 True或 False,fil ...
- python中filter()函数
filter()函数是 Python 内置的另一个有用的高阶函数,filter()函数接收一个函数 f 和一个list,这个函数 f 的作用是对每个元素进行判断,返回 True或 False,filt ...
- 2.python函数编程-filter函数
fileter功能主要使用在需要对数据进行多种操作,并对数据进行过滤的操作. 普通函数实现: movie = ['sb_alex', 'wupei', 'tiger', 'goosb','xxfd', ...
- Python之filter函数
描述 filter() 函数用于过滤序列,过滤掉不符合条件的元素,返回由符合条件元素组成的新列表. 该接收两个参数,第一个为函数,第二个为序列,序列的每个元素作为参数传递给函数进行判,然后返回 Tru ...
- python的filter函数的使用方法详解以及使用案例,是否以什么结尾,是否大于什么(判断是True,则留下来)
1.总共有3个人看电影,有2个人看电影经常说话,我们把他们两个过滤出去 move_people=["gouguoqi","beiye_sb","xiu ...
- Python之filter()函数与替代实现
介绍 filter(f,x)函数用于过滤序列并返回迭代器,结果保留x中f为True的元素,需要新的序列通过list()转换. 例子 过滤列表中的字符串,保留数字. >>> i = [ ...
- Python 中filter函数用法
filter()和map一样,接收一个函数和一个序列.和map不同的是,filter()把传入的函数依次作用于每个元素,然后根据返回值是True还是False决定保留还是丢弃该元素 过滤出奇数: de ...
随机推荐
- vue2.0 子组件props接受父组件传递的值,能不能修改的问题整理
父组件代码: <!-- --> <template> <div class=''> <el-link type="danger">传 ...
- vue router 报错: Uncaught (in promise) NavigationDuplicated {_name:""NavigationDuplicated"... 的解决方法
参考资料:https://blog.csdn.net/zy21131437/article/details/99548983
- MySQL: 1006 - Can't create database '***' (errno: 13) 错误 解决方法
原文连接:https://blog.csdn.net/kexiaoling/article/details/50259569 如果使用root账号登录到数据库create database时提错错误: ...
- [Agc036C]Triangle_数学
Triangle 题目链接:https://atcoder.jp/contests/agc036/tasks/agc036_a 题解: 我开始的时候以为是$Millar-Rabin$加$Pollard ...
- 【转帖】Office的光荣历史(2)
Office的光荣历史(2) https://www.sohu.com/a/201411215_657550 2017-10-31 10:57 7.MS Office 2000 (Office 9.0 ...
- Python time strptime()方法
Python time strptime()方法 描述 Python time strptime() 函数根据指定的格式把一个时间字符串解析为时间元组. 语法 strptime()方法语法: time ...
- Linux就该这么学——新手必须掌握的命令之常用的系统工作命令
echo命令 含义:echo命令用于在终端输出字符串或变量提取后的值,格式为 : echo [字符串|$变量] 示例: 将”Linuxprobe.com”输出到终端屏幕的命令为: [root@linu ...
- java如何防止反编译(转)
出处: java如何防止反编译 一些防止java代码被反编译的方法 综述(写在前面的废话) Java从诞生以来,其基因就是开放精神,也正因此,其可以得到广泛爱好者的支持和奉献,最终很快发展壮大,以至于 ...
- T100弹出是否确认窗体方式
例如: IF NOT cl_ask_confirm('aim-00108') THEN CALL s_transaction_end(') CALL cl_err_collect_show() RET ...
- vue axios拦截跳转
第一步:添加需要拦截的页面 { path: '/control', name: 'Control', meta: { requireAuth: true }, 第二步:页面拦截 router.befo ...