Векторизация вычисления - вектор Браве между парами точек - PullRequest
0 голосов
/ 28 июня 2019

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

Я имею дело с n-массивом (mean_pos) формы (2, 91), и я выполнял указанное вычисление, используя циклы for.Поскольку циклы в Python медленны (и это не простой способ работы), я пытаюсь векторизовать код.

С помощью циклов for:

def bravais_vector(natoms_i, mean_pos):
   b_matrix = []
   b_matrix_row = []

   lx = mean_pos[0].max() - mean_pos[0].min()
   ly = mean_pos[1].max() - mean_pos[1].min()

   for i in range(natoms_i):
       for j in range(natoms_i):

           dist_ij_x = mean_pos[0][i] - mean_pos[0][j]
           dist_ij_y = mean_pos[1][i] - mean_pos[1][j]

           if dist_ij_x > lx/2:
                dist_ij_x = -(lx - dist_ij_x)

           if dist_ij_y > ly/2:
                dist_ij_y = - (ly - dist_ij_y)

           if dist_ij_x < -lx/2:
               dist_ij_x = (lx + dist_ij_x)

           if dist_ij_y < -ly/2:
               dist_ij_y =  (ly + dist_ij_y)

           a2_opt = 2/np.sqrt(3) * dist_ij_y
           a1_opt = dist_ij_x - 0.5 * a2_opt

           b_matrix_row.append(np.array([ np.rint(a1_opt), np.rint(a2_opt) ]))

       b_matrix.append(b_matrix_row)
       b_matrix_row = []
   return np.array(b_matrix)

Векторизация:

def bravais_vector(natoms_i, mean_pos):

    b_matrix = []
    b_matrix_row = []

    lx = mean_pos[0].max() - mean_pos[0].min()
    ly = mean_pos[1].max() - mean_pos[1].min()

    mean_pos_x = np.reshape(mean_pos[0], (len(mean_pos[0]),1))
    mean_pos_y = np.reshape(mean_pos[1], (len(mean_pos[1]),1))

    tiled_mean_pos_x = np.tile(np.transpose(mean_pos_x), (len(mean_pos_x) , 1))
    tiled_mean_pos_y = np.tile(np.transpose(mean_pos_y), (len(mean_pos_y) , 1))

    dist_ij_x = mean_pos_x - tiled_mean_pos_x
    dist_ij_y = mean_pos_y - tiled_mean_pos_y

    dist_ij_x = np.where(dist_ij_x > lx/2, -(lx - dist_ij_x), dist_ij_x)
    dist_ij_y = np.where(dist_ij_y > ly/2, -(ly - dist_ij_y), dist_ij_y)

    dist_ij_x = np.where(dist_ij_x < -lx/2, lx + dist_ij_x, dist_ij_x)
    dist_ij_y = np.where(dist_ij_y < -ly/2, ly + dist_ij_y, dist_ij_y)


    a2_opt = np.rint(np.multiply(2 / (np.sqrt(3)), dist_ij_x))
    a1_opt = np.rint(dist_ij_x - np.multiply(0.5, a2_opt))

    return np.stack((a1_opt, a2_opt), axis=2)

1 Ответ

0 голосов
/ 28 июня 2019

Будьте осторожны, потому что в векторизованной версии вы написали:

a2_opt = np.rint(np.multiply(2 / (np.sqrt(3)), dist_ij_x))

вместо (согласно первой версии)

a2_opt = np.rint(np.multiply(2 / (np.sqrt(3)), dist_ij_y))

Надеюсь, это поможет.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...