在Python中查找与某个字符串相关的所有元组

2024-04-25 05:40:22 发布

您现在位置:Python中文网/ 问答频道 /正文

我试图找到与字符串相关的所有元组,而不仅仅是与之匹配的元组。 这是我做的:

from itertools import chain

data = [('A','B'),('B','C'),('B','D'),('B','F'),('F','W'),('W','H'),('G','Z')]
init = 'A'

filtered_init = [item for item in data if item[0] == init or item[1] == init]
elements = list(dict.fromkeys([ i for i in chain(*filtered_init)]))
elements.remove(init)

dat = []
for i in elements:
    sync = [item for item in data if item[0] == i or item[1] == i]
    dat.append(sync)

print(dat)

结果是:

[('A', 'B'), ('B', 'C'), ('B', 'D'), ('B', 'F')]

但是,它只包含与A-B相关的级别。 我想找到的是与init字符串相关的所有元组,如下图所示:

enter image description here

换句话说,[('A','B'),('B','C'),('B','D'),('B','F'),('F','W'),('W','H')] 它是找到所有的边可以到达init。 我怎样才能得到它们?你知道吗


Tags: or字符串infromchainfordataif
3条回答

为了提高效率,可以使用DSU。此解决方案适用于O(N)

from functools import reduce
import random

parent = dict()
init = 'A'
data = [('A','B'),('B','C'),('B','D'),('B','F'),('F','W'),('W','H'),('G','Z')]

def make_set(v):
    parent[v] = v

def find_set(v):
    if v == parent[v]:
        return v
    parent[v] = find_set(parent[v])
    return parent[v]

def union_sets(a, b):
    a, b = map(find_set, [a, b])
    if a != b:
        if random.randint(0, 1):
            a, b = b, a
        parent[b] = a;

elements = set(reduce(lambda x, y: x+y, data))

for v in elements:
    parent[v] = v

for u, v in data:
    union_sets(u, v)

init_set = find_set(init)
edges_in_answer = [e for e in data if find_set(e[0]) == init_set]
print(edges_in_answer)

输出:[('A','B'),('B','C'),('B','D'),('B','F'),('F','W'),('W','H')]

您的问题是在由edge list data structure定义的无向图中找到initconnected component。你知道吗

对于这个问题,使用这个数据结构不是很方便,所以第一步是将它转换成adjacency list。从那里,我们可以应用任何标准的graph traversal算法,比如depth first search。完成后,我们可以将结果转换回您想要的输出边缘列表格式。你知道吗

from collections import defaultdict

def find_connected_component(edge_list, start):
    # convert to adjacency list
    edges = defaultdict(list)
    for a, b in edge_list:
        edges[a].append(b)
        edges[b].append(a)

    # depth-first search
    stack = [start]
    seen = set()

    while stack:
        node = stack.pop()
        if node not in seen:
            seen.add(node)
            stack.extend(edges[node])

    # convert back to edge list
    return [ edge for edge in edge_list if edge[0] in seen ]

用法:

>>> find_connected_component(data, init)
[('A', 'B'), ('B', 'C'), ('B', 'D'), ('B', 'F'), ('F', 'W'), ('W', 'H')]

一个非常幼稚的解决方案,对于复杂的树可能不太有效。你知道吗

data = [('A', 'B'), ('B', 'C'), ('B', 'D'), ('B', 'F'),
        ('F', 'W'), ('W', 'H'), ('G', 'Z')]
init = ['A']
result = []
while init:
    initNEW = init.copy()
    init = []
    new = 0
    for edge in data:
        for vertex in initNEW:
            if edge[0] == vertex:
                result.append(edge)
                init.append(edge[1])
                new += 1
    for i in range(len(result) - new, len(result)):
        data.remove(result[i])
print(result)
# [('A', 'B'), ('B', 'C'), ('B', 'D'), ('B', 'F'), ('F', 'W'), ('W', 'H')]

相关问题 更多 >