我正在尝试编写将使用numpy和scipy生成视差贴图的代码,但是我在我的numpy数组中存储的值与我的输出图像中实际显示的值完全不同,已保存与misc.imsave。例如,在数组中,没有一个值大于22,但在图像中,我有一个从0到255的全范围值。我认为也许imsave正在拉伸值,以便最大值显示为图像中有255个,但我有其他使用imsave创建的图像,其最大值低于255。
这些是我用来创建视差贴图的函数,给定了两个沿x轴移动的pgm图像:
def disp(i, j, winSize, leftIm, rightIm): #calculate disparity for a given point
width = leftIm.shape[1]
height = leftIm.shape[0]
w = winSize / 2
minSAD = 9223372036854775807 #max int
for d in range(23):
SAD = 0.0 #SAD
k = i - w
v = i + w
m = j - w
n = j + w
for p in range(k, v+1): #window - x
for q in range(m, n+1): #window y
if(p - d > 0 and p < width and q < height):
SAD += abs((int(leftIm[q][p]) - int(rightIm[q][p - d])))
if(SAD < minSAD):
minSAD = SAD
disp = d
# print "%d, %d" % (i, j)
return (disp, SAD)
def dispMap(winSize, leftIm, rightIm):
width = leftIm.shape[1]
height = leftIm.shape[0]
outIm = np.zeros((height, width))
SADstore = np.zeros((height, width))
w = winSize / 2
for i in range(w, width-w):
for j in range(w, height/3-w):
dispout = disp(i, j, winSize, leftIm, rightIm)
outIm[j][i] = 1 * dispout[0] #should normally multiply by 4
SADstore[j][i] = dispout[1]
return (outIm, SADstore)
忽略SAD / SADstore返回值,我确保这些不会影响我当前的进程。
这是我用来获取输出的代码:
disp12 = dispMap(9, view1, view2)
disp12im = disp12[0]
misc.imsave('disp121.pgm', disp12im)
目前的情况是,disp12im中的任何内容都不应该是&gt; 23.如果我运行for循环来检查数组,这仍然是正确的。但是,如果我加载保存的图像并在值上运行相同的循环,我会得到大量的数字超过23.我做错了什么?
答案 0 :(得分:5)
当数组的dtype
从np.float64
(数据类型disp12im
)更改为图像中存储的8位值时,数据会重新调整。
要避免这种情况,请先将图片转换为数据类型np.uint8
,然后再将其提交给imsave
:
misc.imsave('disp121.pgm', disp12im.astype(np.uint8))
例如,我将此x
保存为PGM图像:
In [13]: x
Out[13]:
array([[ 1., 3., 5.],
[ 21., 23., 25.]])
In [14]: x.dtype
Out[14]: dtype('float64')
保存x
不变,然后阅读:
In [15]: imsave('foo.pgm', x)
In [16]: imread('foo.pgm')
Out[16]:
array([[ 0, 21, 42],
[212, 234, 255]], dtype=uint8)
这些值已扩展到完整的8位范围。
相反,请在保存之前将x
转换为np.uint8
,然后再将其读回:
In [17]: imsave('foo.pgm', x.astype(np.uint8))
In [18]: imread('foo.pgm')
Out[18]:
array([[ 1, 3, 5],
[21, 23, 25]], dtype=uint8)