Для оптимизации цикла для создания матрицы смежности

В настоящее время я работаю с графом с помеченными ребрами. Исходная матрица смежности представляет собой матрицу формы [n_nodes, n_nodes, n_edges], где каждая ячейка [i,j, k] равна 1, если узлы i и j соединены ребром k.

Мне нужно создать реверс исходного графа, где узлы становятся ребрами, а ребра становятся узлами, поэтому мне нужна новая матрица с формой [n_edges, n_edges, n_nodes], где каждая ячейка [i,j,k] равна 1, если ребра i и j имеют k как общую вершину.

Для оптимизации цикла для создания матрицы смежности

Следующий код корректно выполняет задачу, но использование 5 вложенных циклов for слишком медленное, чтобы обработать количество графов, с которыми мне приходится работать, кажется, около 700 часов.

Есть ли лучший способ реализовать это?

n_nodes = extended_adj.shape[0]
n_edges = extended_adj.shape[2]
reversed_graph = torch.zeros(n_edges, n_edges, n_nodes, 1)
for i in range(n_nodes):
    for j in range(n_nodes):
        for k in range(n_edges):
            #If adj_mat[i][j][k] == 1 nodes i and j are connected with edge k
            #For this reason the edge k must be connected via node j to every outcoming edge of j
            if extended_adj[i][j][k] == 1:
                #Given node j, we need to loop through every other possible node (l)
                for l in range(n_nodes):
                    #For every other node, we need to check if they are connected by an edge (m)
                    for m in range(n_edges):
                        if extended_adj[j][l][m] == 1:
                            reversed_graph[k][m][j] = 1

Спасибо заранее.

Может быть, это могло бы помочь: «Матрица смежности ребер графа G идентична матрице смежности вершин линейного графа L (G) графа G»

Boschi Francesco 22.03.2022 10:07

Пхе, только сейчас понял твой предыдущий (удаленный) вопрос. Там вы могли бы получить лучшую обратную связь, если бы сказали, что рассматриваемый тензор предназначен для представления графов.

Alexey Birukov 22.03.2022 10:40

@AlexeyBirukov удаленный вопрос был предназначен для объединения графиков, и я его решил, это еще одна проблема.

Boschi Francesco 22.03.2022 10:46

Да я получил его. Вы зря удалили свой вопрос, лучше бы вы дополнили информацию и предоставили решение.

Alexey Birukov 22.03.2022 10:50

Что касается нового, [n_edges, n_edges, n_узлов] скорее будет [n_edges, n_edges, new_edges], не так ли?

Alexey Birukov 22.03.2022 10:56

@AlexeyBirukov каждая пара ребер может иметь общий один из исходных узлов, и мне нужно отслеживать, какой из них их соединяет, поэтому new_edges равен n_nodes

Boschi Francesco 22.03.2022 10:59

Это представление графа ужасно неэффективно. Почему бы вам не использовать квадратную матрицу, в которой элементы являются номером ребра? (И зарезервированное значение для «не края»)

Yves Daoust 22.03.2022 15:12
Почему в Python есть оператор "pass"?
Почему в Python есть оператор "pass"?
Оператор pass в Python - это простая концепция, которую могут быстро освоить даже новички без опыта программирования.
Некоторые методы, о которых вы не знали, что они существуют в Python
Некоторые методы, о которых вы не знали, что они существуют в Python
Python - самый известный и самый простой в изучении язык в наши дни. Имея широкий спектр применения в области машинного обучения, Data Science,...
Основы Python Часть I
Основы Python Часть I
Вы когда-нибудь задумывались, почему в программах на Python вы видите приведенный ниже код?
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
LeetCode - 1579. Удаление максимального числа ребер для сохранения полной проходимости графа
Алиса и Боб имеют неориентированный граф из n узлов и трех типов ребер:
Оптимизация кода с помощью тернарного оператора Python
Оптимизация кода с помощью тернарного оператора Python
И последнее, что мы хотели бы показать вам, прежде чем двигаться дальше, это
Советы по эффективной веб-разработке с помощью Python
Советы по эффективной веб-разработке с помощью Python
Как веб-разработчик, Python может стать мощным инструментом для создания эффективных и масштабируемых веб-приложений.
1
7
77
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Вторя комментариям выше, это представление графа почти наверняка громоздко и неэффективно. Но, несмотря на это, давайте определим векторизованное решение без циклов, которое по возможности использует тензорные представления, что должно быть достаточно эффективным для вычислений для больших графов.

Для ясности давайте использовать [i,j,k] для индексации G (исходный график) и [i',j',k'] для индексации G' (новый график). И давайте сократим n_edges до e и n_nodes до n.

Рассмотрим двумерную матрицу slice = torch.max(G,dim = 1). В каждой координате [a,b] этого среза 1 указывает, что узел a соединен ребром b с каким-то другим узлом (нам все равно с каким).

slice = torch.max(G,dim = 1)                                     # dimension [n,e]

Мы уже на пути к решению, но нам нужно выражение, которое говорит нам, связано ли a с ребром b и другим ребром c для всех ребер c. Мы можем отобразить все комбинации b,c, расширив slice, скопировав и переместив их, а также найдя пересечения между ними.

expanded_dim = [slice.shape[0],slice.shape[1],slice.shape[1]]    # value [n,e,e]

# two copies of slice, expanded on different dimensions
expanded_slice = slice.unsqueeze(1).expand(expanded_dim)         # dimension [n,e,e]
transpose_slice = slice.unsqueeze(2).expand(expanded_dim)        # dimension [n,e,e]

G = torch.bitwise_and(expanded_slice,transpose_slice).int()      # dimension [n,e,e]

G[i',j',k'] теперь равно 1 тогда и только тогда, когда узел i' соединен ребром j' с каким-то другим узлом, И узел i' соединен ребром k' с каким-то другим узлом. Если j' = k' значение равно 1, если одна из конечных точек этого ребра равна i'.

Наконец, мы изменяем размеры, чтобы получить желаемую форму.

G = torch.permute(G,(1,2,0))           # dimension [e,e,n]

Другие вопросы по теме