Вы можете изменить предыдущее решение следующим образом:
from operator import mul
import pandas as pd
import networkx as nx
from functools import reduce
data = pd.DataFrame({'shop': ['S1', 'S1', 'S2', 'S2', 'S3'],
'owner': ['S2', 'S3', 'O1', 'O2', 'O1'],
'share': [0.8, 0.2, 0.5, 0.5, 1.0]})
G = nx.from_pandas_edgelist(data,'shop','owner',edge_attr = ('share'),
create_using=nx.DiGraph())
owners = set(data['owner'])
shops = set(data['shop'])
result = []
summary = {}
for shop in shops:
for owner in owners:
for path in nx.all_simple_paths(G, shop, owner):
share = reduce(mul, (G[start][end]['share'] for start, end in zip(path[:-1], path[1:])), 1)
summary[(owner, shop)] = summary.get((owner, shop), 0) + share
summary = pd.DataFrame.from_dict(summary, orient = 'index', columns = 'share'.split())
print(summary)
выход
share
(O2, S2) 0.5
(O2, S1) 0.4
(S3, S1) 0.2
(O1, S2) 0.5
(O1, S3) 1.0
(O1, S1) 0.6
(S2, S1) 0.8
Линия:
share = reduce(mul, (G[start][end]['share'] for start, end in zip(path[:-1], path[1:])), 1)
вычисляет долю для определенного пути. Затем эта доля агрегируется по всем путям с использованием следующей строки:
summary[(owner, shop)] = summary.get((owner, shop), 0) + share