リスト(数列)の差分のリストがほしかったんだけど、はまったのでメモ。
最初、↓こう書いた。
def difference(lis):
d = reduce(lambda x, y: (x[0].append(y-x[1]), y), lis, ([],0))
d[0].pop(0)
return d[0]
a = range(10)
diff = difference(a)
print diff
2行目のx[0].append(y-x[1])で差分を蓄積していってそれを返している(3行目は余分な初期値との差分があるからそれを取り除いている)。
ところが実行するこうだ。
^o^ > python difference.py
Traceback (most recent call last):
File "difference.py", line 9, in
diff = difference(a)
File "difference.py", line 3, in difference
d = reduce(lambda x, y: (x[0].append(y-x[1]), y), lis, ([],0))
File "difference.py", line 3, in
d = reduce(lambda x, y: (x[0].append(y-x[1]), y), lis, ([],0))
AttributeError: 'NoneType' object has no attribute 'append'
‘NoneType’のオブジェクトは’append’なんて属性持ってないと。しばらく悩んだけど、原因はタイトルのとおり。
1回目のappendを呼び出したところでNoneが返ってくるので、2回目の呼び出しでは失敗する。
>>> a = range(5) >>> a [0, 1, 2, 3, 4] >>> type(a.append(5)) <type 'NoneType'>
結局、次のように書き換えたらうまくいったけど、関数がひとつ増えてしまった。
def difference2(lis):
d = reduce(lambda x, y: (add_list(x[0], y-x[1]), y), lis, ([],0))
d[0].pop(0)
return d[0]
def add_list(lis, x):
lis.append(x)
return lis
a = range(10)
diff = difference2(a)
print diff
^o^ > python difference2.py [1, 1, 1, 1, 1, 1, 1, 1, 1]
RubyのArray#pushならselfを返してくれるから素直に書けるんだけどなあ。