|
|
Строка 40: |
Строка 40: |
| | | |
| === Псевдокод === | | === Псевдокод === |
− | '''dfs'''(<tex> v </tex>) | + | '''function''' dfs(v) |
− | <tex> time = time + 1</tex> | + | time = time + 1 |
− | <tex>enter[v] = time</tex> | + | enter[v] = time |
− | <tex>ret[v] = time </tex> | + | ret[v] = time |
− | '''for''' всех <tex>u</tex> смежных с <tex>v</tex> | + | '''for''' всех u смежных с v |
− | ''if'' <tex>(v, u)</tex> — обратное ребро | + | '''if''' (v, u) — обратное ребро |
− | <tex>ret[v] = \min(ret[v], enter[u])</tex> | + | ret[v] = <tex>\min</tex>(ret[v], enter[u]) |
− | '''if''' вершина <tex>u</tex> — белая | + | '''if''' вершина u — белая |
− | '''dfs'''(u) | + | dfs(u) |
− | <tex> ret[v] = \min(ret[v], ret[u]) </tex> | + | ret[v] = <tex>\min</tex>(ret[v], ret[u]) |
− | '''if''' <tex>ret[u] > enter[v]</tex> | + | '''if''' ret[u] > enter[v] |
− | ребро <tex>(v, u)</tex> — мост | + | ребро (v, u) — мост |
| | | |
| ==См. также== | | ==См. также== |
Версия 13:26, 4 января 2016
Дан неориентированный граф [math] G [/math]. Найти все мосты в [math] G [/math] за время [math] O(|V| + |E|)[/math]
Алгоритм
Теорема: |
Пусть [math] T [/math] — дерево обхода в глубину графа [math] G[/math]. Ребро [math] (u, v) [/math] является мостом тогда и только тогда, когда [math] (u, v) \in T[/math] и из вершины [math] v[/math] и любого ее потомка нет обратного ребра в вершину [math] u[/math] или предка [math] u [/math] |
Доказательство: |
[math]\triangleright[/math] |
[math] \Leftarrow[/math]
Удалим [math] (u, v)[/math] из [math] G[/math]. Докажем, что мы не сможем достичь ни одного из предков [math] v [/math] (в частности [math] u [/math]). Докажем этот факт от противного.
Пусть это не так, и [math] w[/math] — предпоследняя вершина на пути от [math] v[/math] до ее предка [math]x [/math]. Очевидно, [math] (w, x)[/math] не ребро дерева (в силу единственности пути в дереве). Если [math] (w, x)[/math] — обратное ребро, то это противоречит условию теоремы, так как [math] x[/math] — предок [math] u[/math]. Следовательно мы не достигнем предков [math]v[/math], а значит количество компонент связности увеличилось, поэтому ребро [math](u, v)[/math] является мостом.
[math] \Rightarrow[/math]
Пусть существует удовлетворяющее условию обратное ребро [math](x, w)[/math]. Тогда [math](u, v)[/math] лежит на цикле [math]x \rightsquigarrow v \rightarrow u \rightsquigarrow w \rightarrow x[/math] и не может быть мостом. |
[math]\triangleleft[/math] |
Функция [math]ret(v)[/math]
Определим функцию [math]ret(v)[/math], где [math]v \in V[/math], как минимум из следущих величин
- [math]enter(v)[/math] время входа в вершину [math]v [/math]
- [math]enter(x)[/math], где [math]x[/math] — потомок [math]v[/math]
- [math]enter(x)[/math], где [math](w, x)[/math] — обратное ребро, а [math]w[/math] — потомок [math]v[/math] (в нестрогом смысле)
Лемма
Лемма: |
Ребро [math](u, v)[/math] является мостом тогда и только тогда, когда [math](u, v)[/math] принадлежит дереву обхода в глубину и [math]ret(v) \gt enter(u)[/math] |
Доказательство: |
[math]\triangleright[/math] |
Рассмотрим вершину [math]v[/math] или её потомка. Из нее есть обратное ребро в предка [math]v[/math] тогда и только тогда, когда найдется такой сын [math]t[/math], что [math]ret[t] \le enter[v][/math]. Если [math]ret[t] = enter[v][/math], то найдется обратное ребро, приходящее точно в [math]v[/math]. Если же [math]ret[t] \lt enter[v][/math], то это означает наличие обратного ребра в какого-либо предка вершины [math]v[/math].
Таким образом, если для текущего ребра [math](v, t)[/math] (принадлежащего дереву поиска) выполняется [math]ret[t] \gt enter[v][/math], то это ребро является мостом; в противном случае оно мостом не является. |
[math]\triangleleft[/math] |
Утверждение: |
[math]ret(v)[/math] = [math]\min([/math] [math]enter(v) [/math], [math]enter(p)[/math], [math]ret(u)[/math]
[math]) [/math], где [math](v, p)[/math] — обратное ребро, [math](v, u)[/math] — ребро дерева |
[math]\triangleright[/math] |
В скобах у вершины [math]u[/math] указаны [math]enter[u][/math] и [math]ret[u][/math]. Мостами будут красные ребра
- [math]enter(v) [/math]
По определению функции [math]ret[/math]
- [math]enter(p)[/math], [math](v, p)[/math] — обратное ребро
[math]p[/math] достижима из [math]v[/math] по одному обратному ребру, значит величина [math]ret(v)[/math] не больше [math]enter(p)[/math]
- [math]ret(u)[/math], [math]u[/math] — потомок [math]v[/math]
Так как вершина [math]u[/math] — потомок [math]v[/math], то обратное ребро из ее поддерева является обратным ребром из поддерева [math]v[/math]
|
[math]\triangleleft[/math] |
Псевдокод
function dfs(v)
time = time + 1
enter[v] = time
ret[v] = time
for всех u смежных с v
if (v, u) — обратное ребро
ret[v] = [math]\min[/math](ret[v], enter[u])
if вершина u — белая
dfs(u)
ret[v] = [math]\min[/math](ret[v], ret[u])
if ret[u] > enter[v]
ребро (v, u) — мост
См. также
Источники информации