考虑以下 Python 片段:
af=open("a",'r')
bf=open("b", 'w')
for i, line in enumerate(af):
if i < K:
bf.write(line)
现在,假设我要处理 is 的情况K
,None
所以写入继续到文件的末尾。我目前在做
if K is None:
for i, line in enumerate(af):
bf.write(line)
else:
for i, line in enumerate(af):
bf.write(line)
if i==K:
break
这显然不是处理此问题的最佳方法,因为我正在复制代码。有没有更综合的方法可以处理这个问题?很自然的事情是只有在is notif/break
时才存在代码,但这涉及像 Lisp 宏那样动态编写语法,而 Python 不能真正做到这一点。为了清楚起见,我并不关心特定案例(我选择它的部分原因是为了它的简单性),而是学习我可能不熟悉的一般技术。K
None
更新:阅读人们发布的答案并进行更多实验后,这里有更多评论。
如上所述,我一直在寻找可以推广的通用技术,我认为@Paul 的答案,即使用takewhile
from iterrools
,最适合。作为奖励,它也比我上面列出的简单方法快得多;我不确定为什么。itertools
虽然我看过几次,但我不是很熟悉。从我的角度来看,这是For The Win的函数式编程案例!(有趣的是,作者itertools
曾经询问过有关 drop 的反馈。请参阅以http://mail.python.org/pipermail/python-list/2007-December/522529.htmltakewhile
开头的线程。)我在上面简化了我的情况,实际情况有点混乱 - 我正在循环中写入两个不同的文件。所以代码看起来更像:
for i, line in enumerate(af):
if i < K:
bf.write(line)
cf.write(line.split(',')[0].strip('"')+'\n')
鉴于我发布的示例,@Jeff 合理地建议在 when is 的情况下K
,None
我只需复制文件。因为在实践中我无论如何都在循环,这样做并不是一个明确的选择。但是,takewhile
可以轻松地推广到这种情况。我还有另一个在这里没有提到的用例,但也可以在takewhile
那里使用,这很好。第二个例子看起来像(逐字)
i=0
for line in takewhile(illuminacond, af):
line_split=line.split(',')
pid=line_split[1][0:3]
out = line_split[1] + ',' + line_split[2] + ',' + line_split[3][1] + line_split[3][3] + ',' \
+ line_split[15] + ',' + line_split[9] + ',' + line_split[10]
if pid!='cnv' and pid!='hCV' and pid!='cnv':
i = i+1
of.write(out.strip('"')+'\n')
tf.write(line)
在这里我可以使用条件
if K is None:
illuminacond = lambda x: x.split(',')[0] != '[Controls]'
else:
illuminacond = lambda x: x.split(',')[0] != '[Controls]' and i < K
根据@Paul 的原始示例。但是,尽管代码有效,但我对我i
从外部范围获得的事实并不完全满意。有没有更好的方法来做到这一点?或者也许它应该是一个单独的问题。无论如何,感谢所有回答我问题的人。对@Jeff 的荣誉提及,他提出了一些很好的建议。