Вот один с argmax
для эффективности -
def get3Dboundaries(arr):
row_start = arr.any(2).argmax(1)
row_end = arr.shape[1]-arr.any(2)[:,::-1].argmax(1)-1
col_start = arr.any(1).argmax(1)
col_end = arr.shape[2]-arr.any(1)[:,::-1].argmax(1)-1
return np.c_[row_start,row_end,col_start,col_end]
Пример выполнения -
In [61]: arr
Out[61]:
array([[[0, 0, 0, 0, 0],
[0, 1, 1, 1, 0],
[0, 1, 0, 1, 0],
[0, 1, 0, 0, 0],
[0, 0, 0, 0, 0]],
[[0, 0, 0, 0, 0], # different second slice for variety
[1, 1, 1, 1, 0],
[0, 1, 0, 1, 0],
[0, 0, 0, 1, 0],
[0, 0, 0, 1, 0]]])
In [62]: get3Dboundaries(arr)
Out[62]:
array([[1, 3, 1, 3],
[1, 4, 0, 3]])
Мы можем сделать так, чтобы он обрабатывал все регистры нулей с недопустимым спецификатором, скажем -1
Примерно так:
def get3Dboundaries_v2(arr):
row_start = arr.any(2).argmax(1)
row_end = arr.shape[1]-arr.any(2)[:,::-1].argmax(1)-1
col_start = arr.any(1).argmax(1)
col_end = arr.shape[2]-arr.any(1)[:,::-1].argmax(1)-1
out = np.c_[row_start,row_end,col_start,col_end]
return np.where(arr.any((1,2))[:,None],out,-1)
Пробный прогон -
In [76]: arr
Out[76]:
array([[[0, 0, 0, 0, 0],
[0, 1, 1, 1, 0],
[0, 1, 0, 1, 0],
[0, 1, 0, 0, 0],
[0, 0, 0, 0, 0]],
[[0, 0, 0, 0, 0],
[0, 0, 0, 0, 0],
[0, 0, 0, 0, 0],
[0, 0, 0, 0, 0],
[0, 0, 0, 0, 0]]])
In [78]: get3Dboundaries_v2(arr)
Out[78]:
array([[ 1, 3, 1, 3],
[-1, -1, -1, -1]])