Как я могу избежать использования цикла в этом конкретном фрагменте кода Python? - PullRequest
1 голос
/ 29 апреля 2019

У меня есть специфическая проблема с питоном, которую нужно отчаянно ускорить, избегая использования цикла, но я не знаю, как это сделать.Мне нужно прочитать в подгонке изображения, преобразовать его в массив Numpy (примерно, 2000 х 2000 элементов в размере), а затем для каждого элемента вычислить статистику кольца элементов вокруг него.Теперь, когда у меня есть свой код, статистика кольца вокруг элемента вычисляется с помощью маски.Это быстро, но, конечно, я вызываю эту функцию 2000x2000 раз (медленная часть).Я относительно новичок в питоне.Я думаю, что использовать функцию маски - умно, но я не могу найти способ индивидуально обратиться к каждому элементу.Спасибо за любую помощь, которую вы можете предоставить.

 # First, the function computing the statistics within a ring 
 around the central pixel:<br/>
 # flux       = image intensity at pixel (i,j)<br/>
 # rad1, rad2 = inner and outer radii<br/>
 # array      = image array<br/>_

def snr(flux, i, j, rad1, rad2, array):

     a, b   = i, j
     nx, ny = array.shape
     y, x   = np.ogrid[-a:nx-a, -b:ny-b]

     mask   = (x*x + y*y >= rad1*rad1) & (x*x + y*y <= rad2*rad2)

     Nmask  = np.count_nonzero(mask)

     noise  = 0.6052697 * abs(Nmask * flux - sum(array[mask]))

     return noise



 # Now, the call to snr for each pixel in the array data1:<br/>_


     frame1  = fits.open(in_frame, mode='readonly') # read in fits file            
     data1   = frame1[ext].data                     # convert to np array
     ny, nx  = data1.shape                          # array dimensions                                  


     noise1  = zeros((ny, nx), float)               # empty array                       

     r1 = 5                                     # inner radius (pixels)
     r2 = 7                                     # outer radius (pixels)


     # The function is fast, but calling it 2k x 2k times is not:

     for j in range(ny):
          for i in range(nx):

                  noise1[i,j] = der_snr(data1[i,j], i, j, r1, r2, data1)

1 Ответ

0 голосов
/ 29 апреля 2019

Операция, которую вы пытаетесь выполнить, может быть выражена как свертка изображения . Попробуйте что-то вроде этого:

import numpy as np
import scipy.ndimage
from astropy.io import fits

def make_kernel(inner_radius, outer_radius):
    if inner_radius > outer_radius:
        raise ValueError
    x, y = np.ogrid[-outer_radius:outer_radius + 1, -outer_radius:outer_radius + 1]
    r2 = x * x + y * y
    kernel = (r2 >= inner_radius * inner_radius) & (r2 <= outer_radius * outer_radius)
    return kernel

in_frame = '<file path>'
ext = '...'
frame1 = fits.open(in_frame, mode='readonly')
data1 = frame1[ext].data
inner_radius = 5
outer_radius = 7
kernel = make_kernel(inner_radius, outer_radius)
n_kernel = np.count_nonzero(kernel)
conv = scipy.ndimage.convolve(data1, kernel, mode='constant')
noise1 = 0.6052697 * np.abs(n_kernel * data1 - conv)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...