如果您不介意由于四舍五入而出现略微不同的值,我可以为您压缩得非常好。
from math import pi, sin
interval=2*pi/1024
sinval=lambda i:int(round(sin(i*interval)*36))+50
这里是实际执行你想要的操作的代码;它可以使用。
vals = sorted((sinval(i), i) for i in range(1024))
作为测试数据。如果第一列包含索引,您需要在此处交换
val
和
index
的顺序以进行循环。
ranges, oldval, oldidx = [[0, 0]], 0, 0
for val, index in vals:
if not (val == oldval and index == oldidx + 1):
ranges[-1].append(oldidx)
ranges.append([val, index])
oldval, oldidx = val, index
ranges[-1].append(oldidx)
ranges.pop(0)
ifs = ('if((index >= {1}) and (index <= {2})) return {0};\n'.format(val, start, end)
for val, start, end in ranges)
print ''.join(ifs)
编辑:
哎呀,我漏了一行。已经修复。另外,你的倍增器实际上是36而不是35,我可能在脑海中将(14, 86)四舍五入为(15, 85)。
编辑2:为您展示如何仅存储表格的四分之一。
from math import pi, sin
full = 1024
half = 512
quarter = 256
mag = 72
offset = 50
interval = 2 * pi / full
def sinval(i):
return int(round(sin(i * interval) * (mag // 2))) + offset
vals = [sinval(i) for i in range(quarter)]
def sintable(i):
if i >= half + quarter:
return 2 * offset - vals[full - i - 1]
elif i >= half:
return 2 * offset - vals[i - half]
elif i >= quarter:
return vals[half - i - 1]
else:
return vals[i]
for i in range(full):
assert -1 <= sinval(i) - sintable(i) <= 1
如果你从表中减去偏移量,只需将前两个
-vals[...]
替换即可。
另外,底部的比较有些模糊,因为我会因此出现 72 个 off-by-one 错误。这仅仅是因为你的值被舍入为整数;它们都是处于两个值之间的地方,所以精度几乎没有减少。