是否有类似于F#的Seq.scan()
函数的Python函数?
我想做一些类似于cumsum()
或cumproduct()
的操作,而不需要使用循环。
是否有类似于F#的Seq.scan()
函数的Python函数?
我想做一些类似于cumsum()
或cumproduct()
的操作,而不需要使用循环。
我认为Ignacio的解决方案几乎正确,但需要一个类型为('a -> 'a -> 'a)的运算符,并且不会产生第一个元素。
def scan(f, state, it):
for x in it:
state = f(state, x)
yield state
# test
>>> snoc = lambda xs,x: xs+[x]
>>> list(scan(snoc, [], 'abcd'))
[['a'], ['a', 'b'], ['a', 'b', 'c'], ['a', 'b', 'c', 'd']]
>>> list(scan(operator.add, 0, [1,2,3]))
[1,3,6]
具体来说,Seq.scan
的类型是
('State -> 'T -> 'State) -> 'State -> seq<'T> -> seq<'State>
scan
。('State -> 'State -> 'State) -> seq<'State> -> seq<'State>
这是由于Python默认指定了与reduce
相同的类型。
def scan(op, seq):
it = iter(seq)
result = next(it)
for val in it:
result = op(result, val)
yield result
yield result
,否则结果将缺少第一个元素。 - sepp2kSeq.scan<>()
做了一些模糊的解释。 - Ignacio Vazquez-Abrams你可以使用accumulate函数获得相同的功能,这是Python中scan的等效功能:
from itertools import accumulate
nums=[1,2,3,4,5]
nums_added=list(accumulate(nums,lambda x,y:x+y))
print(nums_added)
#prints
# [1,3,6,10,15]
你也可以使用rx(响应式扩展)来使用scan操作符。
这是一个外部库,所以在使用之前需要安装它。
使用rx,你可以像这样做:
from rx import from_, operators as op
nums=[1,2,3,4,5]
source=from_(nums).pipe(op.scan(lambda x,y:x+y))
source.subscribe(lambda num:print(num)
#prints
#1
#3
#6
#10
#15
Rx 不仅可以做这些,还允许您进行响应式编程。要了解更多,请访问:http://reactivex.io
operator.sum
代替lambda x,y: x+y
。 - undefined聚合函数应该使用reduce
而不是map
。
scan(operator.add, [1,2,3,4]) == [1,3,6,10]
。不是map
或reduce
。 - kennytm