带模式功能的块缩减(下采样)3D阵列

2024-06-16 14:18:09 发布

您现在位置:Python中文网/ 问答频道 /正文

我想通过获取原始值中最频繁的值(模式),对3d数组进行下采样。经过一些研究,我在skimage库中找到了block_reduce函数。例如,如果我想取块的平均值,我可以很容易地做到:

from skimage.measure import block_reduce
image = np.arange(4*4*4).reshape(4, 4, 4)
new_image = block_reduce(image, block_size=(2,2,2), func=np.mean, cval=np.mean(grades))

在我的例子中,我想把func参数传递给mode函数。但是,numpy没有模式函数。根据文档,传递的函数应该接受“axis”作为参数。我尝试了一些变通方法,比如编写自己的函数,组合np.uniquenp.argmax,以及将scipy.stats.mode作为函数传递。他们都失败了

我编写了一些嵌套for循环来实现这一点,但对于大型数组来说,它的速度太慢了。有没有一个简单的方法可以做到这一点?我不一定需要使用sci工具包图像库

先谢谢你


Tags: 函数fromimagereducemodenp模式数组
1条回答
网友
1楼 · 发布于 2024-06-16 14:18:09

让我们先假设输入图像形状可被block_size整除,即相应的形状尺寸可被block_size的每个尺寸参数整除

因此,作为预处理,我们需要在输入image上做块,就像这样-

def blockify(image, block_size):
    shp = image.shape
    out_shp = [s//b for s,b in zip(shp, block_size)]
    reshape_shp = np.c_[out_shp,block_size].ravel()
    nC = np.prod(block_size)
    return image.reshape(reshape_shp).transpose(0,2,4,1,3,5).reshape(-1,nC)

接下来,对于mode查找的具体情况,我们将使用^{}argmax-

# https://stackoverflow.com/a/46256361/ @Divakar
def bincount2D_vectorized(a):    
    N = a.max()+1
    a_offs = a + np.arange(a.shape[0])[:,None]*N
    return np.bincount(a_offs.ravel(), minlength=a.shape[0]*N).reshape(-1,N)

out = bincount2D_vectorized(blockify(image, block_size=(2,2,2))).argmax(1)

或者,我们可以使用^{}-

out = mode(blockify(image, block_size=(2,2,2)), axis=1)[0]

最后,如果整除性的初始假设不成立,我们需要使用适当的pad值进行填充。同样,我们可以使用np.pad,作为blockify方法的一部分

相关问题 更多 >