Использование обхода в глубину для поиска мостов — различия между версиями
м (rollbackEdits.php mass rollback) |
|||
| (не показано 14 промежуточных версий 7 участников) | |||
| Строка 1: | Строка 1: | ||
| − | |||
Дан неориентированный [[Основные определения теории графов#Граф| граф]] <tex> G </tex>. Найти все [[Мост, эквивалентные определения | мосты]] в <tex> G </tex> за время <tex> O(|V| + |E|)</tex> | Дан неориентированный [[Основные определения теории графов#Граф| граф]] <tex> G </tex>. Найти все [[Мост, эквивалентные определения | мосты]] в <tex> G </tex> за время <tex> O(|V| + |E|)</tex> | ||
| Строка 18: | Строка 17: | ||
* <tex>enter(v)</tex> [[Использование обхода в глубину для топологической сортировки | время входа в вершину <tex>v </tex> ]] <br> | * <tex>enter(v)</tex> [[Использование обхода в глубину для топологической сортировки | время входа в вершину <tex>v </tex> ]] <br> | ||
* <tex>enter(x)</tex>, где <tex>x</tex> — потомок <tex>v</tex> <br> | * <tex>enter(x)</tex>, где <tex>x</tex> — потомок <tex>v</tex> <br> | ||
| − | * <tex>enter( | + | * <tex>enter(w)</tex>, где <tex>(w, x)</tex> — обратное ребро, а <tex>w</tex> — потомок <tex>v</tex> (в нестрогом смысле) <br> |
===Лемма=== | ===Лемма=== | ||
| Строка 31: | Строка 30: | ||
{{Утверждение | {{Утверждение | ||
|statement = | |statement = | ||
| − | <tex>ret(v)</tex> = <tex>min(</tex> | + | <tex>ret(v)</tex> = <tex>\min(</tex> <tex>enter(v) </tex>, <tex>enter(p)</tex>, <tex>ret(u)</tex> |
| − | + | <tex>) </tex>, где <br> <tex>(v, p)</tex> — обратное ребро, <br> <tex>(v, u)</tex> — ребро дерева | |
| − | |||
| − | |||
| − | <tex>) </tex> | ||
|proof = | |proof = | ||
| − | [[Файл: | + | [[Файл:Bridges_dfs.png|300px|thumb|right|В скобах у вершины <tex>u</tex> указаны <tex>enter[u]</tex> и <tex>ret[u]</tex>. Мостами будут красные ребра]] |
| − | + | #<tex>enter(v) </tex> <br>По определению функции <tex>ret</tex> <br> | |
| − | По определению функции <tex>ret</tex> <br> | + | #<tex>enter(p)</tex>, <tex>(v, p)</tex> — обратное ребро <br><tex>p</tex> достижима из <tex>v</tex> по одному обратному ребру, значит величина <tex>ret(v)</tex> не больше <tex>enter(p)</tex> <br> |
| − | + | #<tex>ret(u)</tex>, <tex>u</tex> — потомок <tex>v</tex> <br>Так как вершина <tex>u</tex> — потомок <tex>v</tex>, то обратное ребро из ее поддерева является обратным ребром из поддерева <tex>v</tex> <br> | |
| − | <tex>p</tex> достижима из <tex>v</tex> по одному обратному ребру, значит величина <tex>ret(v)</tex> не больше <tex>enter(p)</tex> <br> | ||
| − | |||
| − | Так как вершина <tex>u</tex> — потомок <tex>v</tex>, то обратное ребро из ее поддерева является обратным ребром из поддерева <tex>v</tex> <br> | ||
}} | }} | ||
=== Псевдокод === | === Псевдокод === | ||
| − | ''' | + | '''function''' dfs(v): |
| − | + | time = time + 1 | |
| − | + | enter[v] = time | |
| − | + | ret[v] = time | |
| − | '''for''' всех | + | '''for''' всех u смежных с v |
| − | ''if'' | + | '''if''' (v, u) — обратное ребро |
| − | + | ret[v] = min(ret[v], enter[u]) | |
| − | '''if''' вершина | + | '''if''' вершина u — белая |
| − | + | dfs(u) | |
| − | + | ret[v] = min(ret[v], ret[u]) | |
| − | '''if''' | + | '''if''' ret[u] > enter[v] |
| − | ребро | + | ребро (v, u) — мост |
| − | ==Источники== | + | ==См. также== |
| − | + | *[[Обход в глубину, цвета вершин]] | |
| − | + | *[[Лемма о белых путях]] | |
| − | + | ==Источники информации== | |
| − | + | * [http://e-maxx.ru/algo/bridge_searching MAXimal :: algo :: Поиск мостов] | |
| − | ''Седжвик Р.'' Фундаментальные алгоритмы на C++. Часть 5: Алгоритмы на графах. Пер. с англ. — СПб.: ООО «ДиаСофтЮП», 2002. — С. 123-128 | + | * [http://en.wikipedia.org/wiki/Bridge_(graph_theory) Wikipedia — Bridge] |
| + | * [http://rain.ifmo.ru/cat/view.php/vis/graph-general/biconnected-components-2005| Визуализация поиска мостов] | ||
| + | * ''Седжвик Р.'' Фундаментальные алгоритмы на C++. Часть 5: Алгоритмы на графах. Пер. с англ. — СПб.: ООО «ДиаСофтЮП», 2002. — С. 123-128 | ||
[[Категория: Алгоритмы и структуры данных]] | [[Категория: Алгоритмы и структуры данных]] | ||
[[Категория: Обход в глубину]] | [[Категория: Обход в глубину]] | ||
Текущая версия на 19:12, 4 сентября 2022
Алгоритм
| Теорема: |
Пусть — дерево обхода в глубину графа . Ребро является мостом тогда и только тогда, когда и из вершины и любого ее потомка нет обратного ребра в вершину или предка |
| Доказательство: |
|
|
Функция
Определим функцию , где , как минимум из следущих величин
- время входа в вершину
- , где — потомок
- , где — обратное ребро, а — потомок (в нестрогом смысле)
Лемма
| Лемма: |
Ребро является мостом тогда и только тогда, когда принадлежит дереву обхода в глубину и |
| Доказательство: |
|
Рассмотрим вершину или её потомка. Из нее есть обратное ребро в предка тогда и только тогда, когда найдется такой сын , что . Если , то найдется обратное ребро, приходящее точно в . Если же , то это означает наличие обратного ребра в какого-либо предка вершины . Таким образом, если для текущего ребра (принадлежащего дереву поиска) выполняется , то это ребро является мостом; в противном случае оно мостом не является. |
| Утверждение: |
= , ,
, где — обратное ребро, — ребро дерева |
|
Псевдокод
function dfs(v):
time = time + 1
enter[v] = time
ret[v] = time
for всех u смежных с v
if (v, u) — обратное ребро
ret[v] = min(ret[v], enter[u])
if вершина u — белая
dfs(u)
ret[v] = min(ret[v], ret[u])
if ret[u] > enter[v]
ребро (v, u) — мост
См. также
Источники информации
- MAXimal :: algo :: Поиск мостов
- Wikipedia — Bridge
- Визуализация поиска мостов
- Седжвик Р. Фундаментальные алгоритмы на C++. Часть 5: Алгоритмы на графах. Пер. с англ. — СПб.: ООО «ДиаСофтЮП», 2002. — С. 123-128