• Nenhum resultado encontrado

AULA 20

N/A
N/A
Protected

Academic year: 2022

Share "AULA 20"

Copied!
162
0
0

Texto

(1)

AULA 20

(2)

Digrafos

Fonte: Force Directed Graph Referências: Directed graphs (SW): slides,vídeo.

(3)

Digrafos

Um digrafo (directed graph) consiste de um

conjunto de vértices (bolas) e um conjunto de arcos (flechas)

Exemplo: representação de um digrafo

b d

f

c a

e

(4)

Arcos

Um arco é um par ordenado de vértices Exemplo: v e w são vértices e v-w é um arco

v d

f

c a

w

(5)

Ponta inicial e final

Para cada arco v-w, o vértice v é a ponta inicial e w é a ponta final

Exemplo: v é ponta inicial e w é ponta final de v-w

v w

a

v

c w

f d

(6)

Arcos anti-paralelos

Dois arcos são anti-paralelos se a ponta inicial de um é ponta final do outro

Exemplo: v-w e w-v são anti-paralelos

w

f d

v

a

c

(7)

Digrafos simétricos

Um digrafo é simétrico se cada um de seus arcos é anti-paralelo a outro

Exemplo: digrafo simétrico

e

f d

b

a

c

(8)

Graus de entrada e saída

grau de entrada de v= no. arcos com ponta final v

grau de saída de v = no. arcos com ponta inicial v Exemplo: v tem grau de entrada 1 e de saída 2

v d

f

c a

e

(9)

Número de arcos

Quantos arcos, no máximo, tem um digrafo com V vértices?

A resposta é V×(V−1) = Θ(V2)

digrafo completo = todo par ordenado de vértices distintos é arco

digrafo denso = tem “muitos” muitos arcos digrafo esparso = tem “poucos” arcos

(10)

Número de arcos

Quantos arcos, no máximo, tem um digrafo com V vértices?

A resposta é V×(V−1) = Θ(V2)

digrafo completo = todo par ordenado de vértices distintos é arco

digrafo denso = tem “muitos” muitos arcos digrafo esparso = tem “poucos” arcos

(11)

Especificação

Digrafos podem ser especificados através de sua lista de arcos

Exemplo:

b d

f

c a

e

d-f b-d a-c b-e e-f a-b

(12)

Grafos

Fonte: Scaling Computation of Graph Structured Data with NScale

Referências: Undirected graphs (SW): slides, vídeo.

(13)

Grafos

Um grafo é um digrafo simétrico Exemplo: um grafo

e

f d

b

a

c

(14)

Grafos

Um grafo é um digrafo simétrico Exemplo: representação usual

e

f d

b

a

c

(15)

Arestas

Uma aresta é um par de arcos anti-paralelos.

Exemplo: b-a e a-b são a mesma aresta

e

f d

b

a

c

(16)

Especificação

Grafos podem ser especificados através de sua lista de arestas

Exemplo:

e

f d

b

a

c

f-d b-d c-a e-b e-f a-b

(17)

Graus de vértices

Em um grafo

grau de v = número de arestas com ponta em v Exemplo: v tem grau 3

e

f d

v

a

c

(18)

Número de arestas

Quantas arestas, no máximo, tem um grafo com V vértices?

A resposta é V×(V−1)/2 = Θ(V2)

grafo completo = todo par não-ordenado de vértices distintos é aresta

(19)

Número de arestas

Quantas arestas, no máximo, tem um grafo com V vértices?

A resposta é V×(V−1)/2 = Θ(V2)

grafo completo = todo par não-ordenado de vértices distintos é aresta

(20)

Digrafos no computador

Fonte: GraphX Programming Guide

(21)

Matriz de adjacência de digrafos

Matriz de adjacência de um digrafo tem linhas e colunas indexadas por vértices:

adj[v][w] = 1 se v-w é um arco adj[v][w] = 0 em caso contrário Exemplo:

0

1

2 3

0 1 2 3 0 0 1 1 0 1 0 0 0 1 2 0 1 0 1 3 0 0 0 0

Consumo de espaço: Θ(V2) fácil de implementar

(22)

Matriz de adjacência de grafos

Matriz de adjacência de um grafo tem linhas e colunas indexadas por vértices:

adj[v][w] = 1 se v-w é um aresta adj[v][w] = 0 em caso contrário Exemplo:

0

1

2 3

0 1 2 3 0 0 1 1 0 1 1 0 1 1 2 1 1 0 1 3 0 1 1 0

Consumo de espaço: Θ(V2) fácil de implementar

(23)

Digrafo

Digraph G

0

2

4

3 5

1

(24)

Estruturas de dados

0 0 0

0 0 0

0 0 0 0

5

5

0

0 0 0 0 0 0

0 0 0 0 0

0 0

1 1 1

1 1 1

1 1

1 1

0

0

1 2 3 4 0

0 1 2 3 4

A

(25)

Digrafos no algs4

(26)

Matriz de incidência de digrafos

Uma matriz de incidências de um digrafo tem linhas indexadas por vértices e colunas por arcos e cada entrada [k][vw] é −1 se k = v, +1 se k = w, e 0 em caso contrário.

0

1

2 3

0-1 0-2 2-1 2-3 1-3

0 -1 -1 0 0 0

1 +1 0 +1 0 -1

2 0 +1 -1 -1 0

3 0 0 0 +1 +1

Consumo de espaço: Θ(nm)

Interessante do ponto de vista de otimização linear.

(27)

Vetor de listas de adjacência de digrafos

Na representação de um digrafo através de listas de adjacência tem-se, para cada vértice v, uma lista dos vértices que são vizinhos v.

Exemplo:

0

1

2 3

0: 1, 2 1: 3 2: 1, 3 3:

Consumo de espaço: Θ(V+A) (linear) Manipulação eficiente

(28)

Vetor de lista de adjacência de grafos

Na representação de um grafo através de listas de adjacência tem-se, para cada vértice v, uma lista dos vértices que são pontas de arestas incidentes a v Exemplo:

0

1

2 3

0: 1, 2 1: 3, 0, 2 2: 1, 3, 0 3: 1, 2

Consumo de espaço: Θ(V+A) (linear) Manipulação eficiente

(29)

Digrafo

Digraph G

0

2

4

3 5

1

(30)

Estruturas de dados

5 5

5

6 10

0 1 2 3

G V

A adj

2 3 4

1 4

4

1

4 1

(31)

Grafos no algs4

(32)

Esqueleto da classe Digraph

public class Digraph {

private int V; // no. vértices private int E; // no. arcos private Bag<Integer>[] adj;

private int[] indegree;

public Digraph(int V) {...}

public int V() { return V; } public int E() { return E; }

public void addEdge(int v, int w) { } public Iterable<Integer> adj(intv) { } public int outdegree(int v) {...}

public int indegree(int v) {...}

public Digraph reverse() { ...}

}

(33)

Digraph

public Digraph(int V) { this.V = V;

this.E = 0;

indegree = new int[V];

adj = (Bag<Integer>[]) new Bag[V];

for (int v = 0; v < V; v++) { adj[v] = new Bag<Integer>();

} }

(34)

Digraph

// insere um arco

public void addEdge(int v, int w) { adj[v].add(w);

indegree[w]++;

E++;

}

// retorna a lista de adjacência de v public Iterable<Integer> adj(int v) {

return adj[v];

}

(35)

Digraph

// retorna o grau de saída de v public int outdegree(int v) {

return adj[v].size();

}

// retorna o grau de entrada de v public int indegree(int v) {

return indegree[v];

}

(36)

Digraph

// retorna o sigrafo reverso public Digraph reverse() {

Digraph reverse = new Digraph(V);

for (int v = 0; v < V; v++) { for (int w : adj(v)) {

reverse.addEdge(w, v);

} }

return reverse;

}

(37)

Caminhos em digrafos

Fonte: Finding Your Way & Making You A Priority

(38)

Caminhos

Um caminho num digrafo é qualquer seqüência da forma v0–v1–v2–...–vk−1–vp, onde vk−1-vk é um arco para k = 1, . . . ,p.

Exemplo: 2-4-1-3-5-4-5 é um caminho com origem 2 é término 5

0

1 3

5

4 2

(39)

Caminhos simples

Um caminho é simples se não tem vértices repetidos

Exemplo: 2-4-1-3-5 é um caminho simples de 2 a 5

0

3

5

4 1

2

(40)

Procurando caminhos

Fonte: Mincecraft maze created by Carl Eklof (algs4)

(41)

Procurando um caminho

Problema: dados um digrafo G e dois vértices s e t decidir se existe um caminho de s a t

Exemplo: para s = 0 e t = 1 a resposta é SIM

0

2

3 5

1

4

(42)

Procurando um caminho

Problema: dados um digrafo G e dois vértices s e t decidir se existe um caminho de s a t

Exemplo: para s = 0 e t = 1 a resposta é SIM

0

2 1

5 3

4

(43)

Procurando um caminho

Problema: dados um digrafo G e dois vértices s e t decidir se existe um caminho de s a t

Exemplo: para s = 5 e t = 4 a resposta é NÃO

0

2

3 5

1

4

(44)

DFSpaths

A classe DFSpaths recebe um digrafo G e um

vértice s e determina todos os vértices alcançáveis a partir de s.

A classe implementa a técnica chamada busca em profundidade (Depth First Search).

public class DFSpaths{

public void DFSpaths(Digraph G,int s){}

// retorna true se há caminho de s a v public boolean hasPath(int v){ ...}

private void dfs(Digraph G, int v) { } }

(45)

DFSpaths(G, 0)

0

2 1

3 5

4

(46)

DFSpaths(G, 0)

0

2 1

3 5

4

(47)

dfs(G, 0)

0

2 1

3 5

4

(48)

dfs(G, 0)

0

2 1

3 5

4

(49)

dfs(G, 2)

0

2 1

3 5

4

(50)

dfs(G, 2)

0

2 1

3 5

4

(51)

dfs(G, 1)

0

2 1

3 5

4

(52)

dfs(G, 2)

0

2 1

3 5

4

(53)

dfs(G, 2)

0

2 1

3 5

4

(54)

dfs(G, 4)

0

2 1

3 5

4

(55)

dfs(G, 4)

0

2 1

3 5

4

(56)

dfs(G, 4)

0

2 1

3 5

4

(57)

dfs(G, 5)

0

2 1

3 5

4

(58)

dfs(G, 5)

0

2 1

3 5

4

(59)

dfs(G, 5)

0

2 1

3 5

4

(60)

dfs(G, 4)

0

2 1

3 5

4

(61)

dfs(G, 2)

0

2 1

3 5

4

(62)

dfs(G, 0)

0

2 1

3 5

4

(63)

dfs(G, 0)

0

2 1

3 5

4

(64)

dfs(G, 3)

0

2 1

3 5

4

(65)

dfs(G, 3)

0

2 1

3 5

4

(66)

dfs(G, 3)

0

2 1

3 5

4

(67)

dfs(G, 3)

0

2 1

3 5

4

(68)

dfs(G, 0)

0

2 1

3 5

4

(69)

dfs(G, 0)

0

2 1

3 5

4

(70)

dfs(G, 0)

0

2 1

3 5

4

(71)

DFSpaths(G, 0)

0

2 1

3 5

4

(72)

DFSpaths(G, 2)

0

2 1

3 5

4

(73)

DFSpaths(G, 2)

0

2 1

3 5

4

(74)

dfs(G, 2)

0

2 1

3 5

4

(75)

dfs(G, 2)

0

2 1

3 5

4

(76)

dfs(G, 1)

0

2 1

3 5

4

(77)

dfs(G, 2)

0

2 1

3 5

4

(78)

dfs(G, 2)

0

2 1

3 5

4

(79)

dfs(G, 4)

0

2 1

3 5

4

(80)

dfs(G, 4)

0

2 1

3 5

4

(81)

dfs(G, 4)

0

2 1

3 5

4

(82)

dfs(G, 5)

0

2 1

3 5

4

(83)

dfs(G, 5)

0

2 1

3 5

4

(84)

dfs(G, 5)

0

2 1

3 5

4

(85)

dfs(G, 4)

0

2 1

3 5

4

(86)

dfs(G, 2)

0

2 1

3 5

4

(87)

DFSpaths(G, 2)

0

2 1

3 5

4

(88)

DFSpaths

public class DFSpaths { private final int s;

private boolean[] marked;

public DFSpaths(Digraph G, int s) {}

private void dfs(Digraph G, int v) {}

public boolean hasPath(int v) {}

}

(89)

DFSpaths

Encontra um caminho de s a todo vértice alcançável a partir de s.

public DFSpaths(Digraph G, int s) { marked = new boolean[G.V()];

this.s = s;

dfs(G, s);

}

(90)

DFSpaths

private void dfs(Digraph G, int v) { marked[v] = true;

for (int w : G.adj(v)) { if (!marked[w]) {

dfs(G, w);

} } }

(91)

DFSpaths

Há um caminho de s a v?

public boolean hasPath(int v) { return marked[v];

}

(92)

DFSpaths(G, 0)

0

2

4

3 5

1

0-2 dfs(G,2) 2-1 dfs(G,1) 2-4 dfs(G,4)

4-1

4-5 dfs(G,5) 5-1

0-3 dfs(G,3) 3-4

3-5 0-4

existe caminho

(93)

DFSpaths(G, 2)

0

2

4

3 5

1

2-1 dfs(G,1) 2-4 dfs(G,4)

4-1

4-5 dfs(G,5) 5-1

nao existe caminho

(94)

Consumo de tempo

Qual é o consumo de tempo de DFSpaths?

Qual é o consumo de tempo da função dfs?

(95)

Consumo de tempo

Qual é o consumo de tempo de DFSpaths?

Qual é o consumo de tempo da função dfs?

(96)

Conclusão

O consumo de tempo de DFSpaths é Θ(V) mais o consumo de tempo da função dfs().

(97)

Conclusão

O consumo de tempo da função dfs() para vetor de listas de adjacência é O(V+E).

O consumo de tempo de DFSpaths para vetor de listas de adjacência é ∼O(V+E).

(98)

Conclusão

O consumo de tempo da função dfs() para matriz de adjacências é O(V2).

O consumo de tempo de DFSpaths para matriz de adjacências é O(V2).

(99)

Caminhos no computador

Fonte: Tron Legacy Light Cycle Riders wallpaper

(100)

Caminhos no computador

Como representar caminhos no computador?

(101)

Caminhos no computador

Uma maneira compacta de representar caminhos de um vértice a outros é uma arborescência

Uma arborescência é um digrafo em que

I existe exatamente um vértice com grau de entrada 0, a raiz da arborescência

I não existem vértices com grau de entrada maior que 1,

I cada um dos vértices é término de um caminho com origem no vértice raiz.

(102)

Arborescências

Exemplo: a raiz da arborescência é 0

0

2 1

3 5

4

(103)

Arborescências

Exemplo: a raiz da arborescência é 0

0

2 1

3 5

4

(104)

Arborescências

Propriedade: para todo vértice v, existe exatamente um caminho da raiz a v

0

2 1

3 5

4

(105)

Arborescências

Todo vértice w, exceto a raiz, tem uma pai: o único vértice v tal que v-w é um arco

0

2 1

3 5

4

(106)

Arborescências no computador

Um arborescência pode ser representada através de um vetor de pais: edgeTo[w] é o pai de w

Se r é a raiz, então edgeTo[r]=r

0

2 1

3 5

4

vértice edgeTo

0 0

1 2

2 0

3 0

4 2

5 4

(107)

Caminho

Dado o vetor de pais, edgeTo, de uma

arborescência, é fácil determinar o caminho que leva da raiz a um dado vértice v: basta inverter a

sequência impressa pelo seguinte fragmento de código:

for (int x=v; edgeTo[x]!=x; x=edgeTo[x]) StdOut.printf("%d-", x);

StdOut.printf("%d", x);

(108)

Caminho

Dado o vetor de pais, edgeTo, de uma

arborescência, é fácil determinar o caminho que leva da raiz a um dado vértice v: basta inverter a

sequência impressa pelo seguinte fragmento de código:

for (int x=v; edgeTo[x]!=x; x=edgeTo[x]) StdOut.printf("%d-", x);

StdOut.printf("%d", x);

(109)

DFSpaths: esqueleto

public class DFSpaths { private final int s;

private boolean[] marked;

private int[] edgeTo;

public DFSpaths(Digraph G, int s) {}

private void dfs(Digraph G, int v) {}

public boolean hasPath(int v) {}

public Iterable<Integer> pathTo(int v) }

(110)

DFSpaths: construtor

Encontra um caminho de s a todo vértice alcançável a partir de s.

public DFSpaths(Digraph G, int s) { marked = new boolean[G.V()];

edgeTo = new int[G.V()];

this.s = s;

dfs(G, s);

}

(111)

DFSpaths: dfs()

private void dfs(Digraph G, int v) { marked[v] = true;

for (int w : G.adj(v)) { if (!marked[w]) {

edgeTo[w] = v;

dfs(G, w);

} } }

(112)

DFSpaths: pathTo()

Retorna um caminho de s a v ou null se um tal caminho não existe.

public Iterable<Integer> pathTo(int v) { if (!hasPath(v)) return null;

Stack<Integer> path =

new Stack<Integer>();

for (int x = v; x != s; x = edgeTo[x]) path.push(x);

path.push(s);

return path;

}

(113)

Busca em largura

Fonte: http://catalog.flatworldknowledge.com/bookhub/

(114)

Busca ou varredura

Um algoritimo de busca (ou varredura) examina, sistematicamente, os vértices e os arcos de um digrafo.

Cada arco é examinado uma só vez.

Depois de visitar sua ponta inicial o algoritmo percorre o arco e visita sua ponta final.

(115)

Busca em largura

A busca em largura (=breadth-first search search

= BFS) começa por um vértice, digamos s, especificado pelo usuário.

O algoritmo visita s,

depois visita vértices à distância 1 de s, depois visita vértices à distância 2 de s, depois visita vértices à distância 3 de s, e assim por diante

(116)

Simulação

i 0 1 2 3 4 5 q[i]

0

2 1

3 5

4

(117)

Simulação

i 0 1 2 3 4 5 q[i]

0

2 1

3 5

4

(118)

Simulação

i 0 1 2 3 4 5 q[i] 0

0

2 1

3 5

4

(119)

Simulação

i 0 1 2 3 4 5 q[i] 0

0

2 1

3 5

4

(120)

Simulação

i 0 1 2 3 4 5 q[i] 0 2

0

2 1

3 5

4

(121)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3

0

2 1

3 5

4

(122)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4

0

2 1

3 5

4

(123)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4

0

2 1

3 5

4

(124)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4

0

2 1

3 5

4

(125)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4

0

2 1

3 5

4

(126)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4

0

2 1

3 5

4

(127)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5

0

2 1

3 5

4

(128)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5

0

2 1

3 5

4

(129)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5

0

2 1

3 5

4

(130)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5

0

2 1

3 5

4

(131)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5 1

0

2 1

3 5

4

(132)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5 1

0

2 1

3 5

4

(133)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5 1

0

2 1

3 5

4

(134)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5 1

0

2 1

3 5

4

(135)

Simulação

i 0 1 2 3 4 5 q[i] 0 2 3 4 5 1

0

2 1

3 5

4

(136)

Arborescência da BFS

A busca em largura a partir de um vértice s descreve a arborescência com raiz s

0

2 1

3 5

4

(137)

Arborescência da BFS

Essa arborescência é conhecida como

arborescência de busca em largura (= BFS tree)

0

2 1

3 5

4

(138)

Representação da BFS

Podemos representar essa arborescência

explicitamente por um vetor de pais edgeTo[]

0

2 1

3 5

4

(139)

Representação da BFS

v 0 1 2 3 4 5 edgeTo 0 5 0 0 0 3

0

2 1

3 5

4

(140)

Class BFSpaths

BFSpaths visita todos os vértices do digrafo G que podem ser alcançados a partir de s.

A visita aos vértices é registrada no vetor

marked[]. Se v foi então marked[v] == true.

Para isso BFSpaths usa uma fila de vértices:

Queue<Integer> q = new Queue<Integer>();

(141)

BFSpaths: esqueleto

public class BFSpaths { private final int s;

private boolean[] marked;

private int[] edgeTo;

public BFSpaths(Digraph G, int s) {}

private void bfs(Digraph G, int s) {}

public boolean hasPath(int v) {}

public Iterable<Integer> pathTo(int v) }

(142)

BFSpaths

Encontra um caminho de s a todo vértice alcançável a partir de s.

public BFSpaths(Digraph G, int s) { marked = new boolean[G.V()];

edgeTo = new int[G.V()];

this.s = s;

bfs(G, s);

}

(143)

bfs(): inicializações

private void bfs(Digraph G, int s) {

Queue<Integer> q= new Queue<Integer>();

marked[v] = true;

q.enqueue(s);

// aqui vem a iteração do próximo slide

(144)

bfs(): iteração

while (!q.isEmpty()) { int v = q.dequeue();

for (int w : G.adj(v)) { if (!marked[w]) {

edgeTo[w] = v;

marked[w] = true;

q.enqueue(w);

} } } }

(145)

BFSpaths

Há um caminho de s a v?

// Método copiado de DFSpaths.

public boolean hasPath(int v) { return marked[v];

}

(146)

BFSpaths

Retorna um caminho de s a v ou null se um tal caminho não existe.

// Método copiado de DFSpaths.

public Iterable<Integer> pathTo(int v) { if (!hasPath(v)) return null;

Stack<Integer> path =

new Stack<Integer>();

for (int x = v; x != s; x = edgeTo[x]) path.push(x);

path.push(s);

return path;

}

(147)

Relações invariantes

Digamos que um vértice v foi visitado se marked[v] == true

No início de cada iteração vale que

I todo vértice que está na fila já foi visitado;

I se um vértice v já foi visitado mas algum de seus vizinhos ainda não foi visitado, então v está na fila.

Cada vértice entra na fila no máximo uma vez.

Portanto, basta que a fila tenha espaço suficiente para G.V() vértices.

(148)

Consumo de tempo

O consumo de tempo da função BFSpaths para vetor de listas de adjacência é O(V+E).

O consumo de tempo da função BFSpathspara matriz de adjacência é O(V2).

(149)

BFS versus DFS

I busca em largura usa fila, busca em profundidade usa pilha

I a busca em largura é descrita em estilo

iterativo, enquanto a busca em profundidade é descrita, usualmente, em estilo recursivo

I busca em largura começa tipicamente num vértice especificado, a busca em

profundidade, o próprio algoritmo escolhe o vértice inicial

I a busca em largura apenas visita os vértices que podem ser atingidos a partir do vértice inicial, a busca em profundidade, tipicamente, visita todos os vértices do digrafo

(150)

Certificados

Achievement Certificate

_________________________________________________________

is presented with the

Computer Achievement Award

On the _______ Day of __________ In the Year _______.

Signed,

____________________________

Certificate Provided by www.hooverwebdesign.com

Fonte: Free Printable Computer Achievement Award Certificates

(151)

Procurando um caminho

Problema: dados um digrafo G e dois vértices s e t decidir se existe um caminho de s a t

Exemplo: para s = 0 e t = 1 a resposta é SIM

0

2

3 5

1

4

(152)

Certificados

Como é possível ’verificar’ a resposta?

Como é possível ’verificar’ que existe caminho?

Como é possível ’verificar’ que não existe caminho?

Veremos questões deste tipo freqüentemente

Elas terão um papel suuupeeer importante no final de MAC0338 Análise de Algoritmos e em MAC0414 Autômatos, Computabilidade e Complexidade

Elas estão relacionadas com o Teorema da

Dualidade visto em MAC0315 Otimização Linear

(153)

Certificados

Como é possível ’verificar’ a resposta?

Como é possível ’verificar’ que existe caminho?

Como é possível ’verificar’ que não existe caminho?

Veremos questões deste tipo freqüentemente

Elas terão um papel suuupeeer importante no final de MAC0338 Análise de Algoritmos e em MAC0414 Autômatos, Computabilidade e Complexidade

Elas estão relacionadas com o Teorema da

Dualidade visto em MAC0315 Otimização Linear

(154)

Certificado de inexistência

Como é possível demonstrar que o problema não tem solução?

0

2

4

3 5

1 dfs(G,2)

2-1 dfs(G,1) 2-4 dfs(G,4)

4-1

4-5 dfs(G,5) 5-1

nao existe caminho

(155)

DFSpath(G,2,3)

0

2 1

3 5

4

(156)

Cortes (= cuts )

Um corte é uma bipartição do conjunto de vértices Um arco pertence ou atravessa um corte (S,T) se tiver uma ponta em S e outra em T

Exemplo 1: arcos em vermelho estão no corte (S,T)

0

2 1

3 5

S

4

T

(157)

Cortes (= cuts )

Um corte é uma bipartição do conjunto de vértices Um arco pertence ou atravessa um corte (S,T) se tiver uma ponta em S e outra em T

Exemplo 2: arcos em vermelho estão no corte (S,T)

0

2 1

3 5

4

S

T

(158)

st-Cortes (= st-cuts )

Um corte (S,T) é um st-corte se s está em S e t está em T

Exemplo: (S,T) é um 1-3-corte um 2-5-corte . . .

0

2 1

3 5

S

4

T

(159)

Certificado de inexistência

Para demonstrarmos que não existe um caminho de s a t basta exibirmos um st-corte (S,T) em que

todo arco no corte tem ponta inicial em T e ponta final em S

(160)

Certificado de inexistência

Exemplo: certificado de que não há caminho de 2 a 3

0

2 1

3 5

4

S

T

(161)

Conclusão

Para quaisquer vértices s e t de um digrafo, vale uma e apenas umas das seguintes afirmações:

I existe um caminho de s a t

I existe st-corte (S,T) em que todo arco no corte tem ponta inicial em T e ponta final em S.

Fonte: Yin and yang (Wikipedia)

(162)

E DFSpaths e BFSpaths com isso?

No código das classes DFSpaths e BFSpaths se existe um caminho de s a t ele esta representado no vetor edgeTo[].

No código da classes DFSpaths e BFSpaths se não existe um caminho de s a t um st-corte separando s de t está representado no vetor marked[].

Em ambos os casos podemos fazer um trecho de código que verifica a resposta em tempo

proporcional a V + E.

Referências

Outline

Documentos relacionados

O desempenho do amplificador híbrido proposto pode ser considerado melhor, com dois lasers de bombeamento Raman e potência de 20 dBm em cada canal WDM, quando a análise é

Significado das medidas e indicadores mais utilizadas Crescimento e desenvolvimento Peso Altura Circunferência da panturrilha Circunferência da cabeça P/Idade Altura /Idade

Este subsídio é para aqueles que forem reformar a casa usando os serviços de alguma construtora que possua endereço na cidade de Hikone.. Informações Setor de Promoção da

A Vila Isabel em Visita Angela Maria Moreira Martin,, Luciana Alves Peterle e Flavia Guimarães

O estômago é um órgão revestido por uma mucosa, e essa mucosa produz enzimas digestivas, protege as células estomacais do ácido clorídrico e faz uma espécie de separação

Este trabalho objetivou com auxílio da fotointerpretação e da análise multivariada analisar os parâmetros dimensionais da rede de drenagem através de 12 microbacias de 3 a ordem

No Programa de Fellow na área de Neurocirurgia, o participante deverá indicar, na carta de interesse e no ato da inscrição, em qual área pretende realizar o aprimoramento -

Caso o arrematante não apresente requerimento de parcelamento no prazo de até 30 (trinta) dias contados da lavratura, nos autos da execução, da certidão de