Skip to content

Commit

Permalink
Merge pull request #25 from dkgoutham/master
Browse files Browse the repository at this point in the history
Implementation of Dinic's Algorithm
  • Loading branch information
jordanmontt authored Nov 2, 2023
2 parents dafd040 + c638689 commit 876fd9c
Show file tree
Hide file tree
Showing 4 changed files with 380 additions and 0 deletions.
51 changes: 51 additions & 0 deletions src/AI-Algorithms-Graph-Components/AIDinicNode.class.st
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
"
This class represents a node in the Dinic's algorithm.
Each node has a level and a currentIndex property.
"
Class {
#name : #AIDinicNode,
#superclass : #AIGraphNode,
#instVars : [
'level',
'currentIndex'
],
#category : #'AI-Algorithms-Graph-Components-Nodes'
}

{ #category : #accessing }
AIDinicNode >> currentIndex [

^ currentIndex
]

{ #category : #setter }
AIDinicNode >> currentIndex: aValue [

currentIndex := aValue
]

{ #category : #initialization }
AIDinicNode >> initialize [

super initialize.
level := -1.
currentIndex := 1.
]

{ #category : #accessing }
AIDinicNode >> label [

^ 'Dinic: '
]

{ #category : #accessing }
AIDinicNode >> level [

^ level
]

{ #category : #setter }
AIDinicNode >> level: aValue [

level := aValue
]
44 changes: 44 additions & 0 deletions src/AI-Algorithms-Graph-Components/AINetworkFLowEdge.class.st
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
"
This class represents an edge in the Dinic's Algorithm.
Each edge has a capacity and a flow property.
"
Class {
#name : #AINetworkFlowEdge,
#superclass : #AIGraphEdge,
#instVars : [
'capacity',
'flow'
],
#category : #'AI-Algorithms-Graph-Components'
}

{ #category : #accessing }
AINetworkFlowEdge >> asTuple [
^{from model. to model. capacity. flow}
]

{ #category : #accessing }
AINetworkFlowEdge >> capacity [
^capacity
]

{ #category : #accessing }
AINetworkFlowEdge >> capacity: aValue [
capacity := aValue
]

{ #category : #accessing }
AINetworkFlowEdge >> flow [
^flow

]

{ #category : #accessing }
AINetworkFlowEdge >> flow: anObject [
flow:=anObject
]

{ #category : #initialization }
AINetworkFlowEdge >> initialize [
flow:=0
]
55 changes: 55 additions & 0 deletions src/AI-Algorithms-Graph-Tests/AIDinicTest.class.st
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
Class {
#name : #AIDinicTest,
#superclass : #TestCase,
#instVars : [
'dinic'
],
#category : #'AI-Algorithms-Graph-Tests'
}

{ #category : #running }
AIDinicTest >> setUp [

super setUp.
dinic := AIDinic new.
]

{ #category : #tests }
AIDinicTest >> testComplexFlow [

| graphType graph value |
graphType := AIWeightedDAGFixture new.
graph := graphType withoutCyclesComplexWeightedGraph.
dinic nodes: graph nodes.
dinic
edges: graph edges
from: [ :each | each first ]
to: [ :each | each second ]
capacity: [ :each | each third ].
dinic
setStartNode: (dinic findNode: $b)
sinkNode: (dinic findNode: $r).

value := dinic run.
self assert: value equals: 2.
]

{ #category : #tests }
AIDinicTest >> testSimpleFlow [

| graphType graph value |
graphType := AIWeightedDAGFixture new.
graph := graphType weightedDAG.
dinic nodes: graph nodes.
dinic
edges: graph edges
from: [ :each | each first ]
to: [ :each | each second ]
capacity: [ :each | each third ].
dinic
setStartNode: (dinic findNode: $B)
sinkNode: (dinic findNode: $E).

value := dinic run.
self assert: value equals: 17.
]
230 changes: 230 additions & 0 deletions src/AI-Algorithms-Graph/AIDinic.class.st
Original file line number Diff line number Diff line change
@@ -0,0 +1,230 @@
"
Dinic's algorithm is a graph algorithm used to solve the maximum flow problem efficiently. It is an improvement over the Ford-Fulkerson algorithm and is based on the concept of layering in the residual graph.
Here's a summary of Dinic's algorithm:
- Start with an initial flow of zero and create the residual graph from the given network.
- Construct a level graph using breadth-first search (BFS) on the residual graph. The level graph assigns levels or distances to each vertex, indicating the shortest path from the source.
- While there exists an augmenting path (a path from the source to the sink in the level graph), do the following steps:
a. Use depth-first search (DFS) to find a blocking flow along the augmenting path. This flow is the maximum amount of flow that can be sent through the path.
b. Update the residual graph by subtracting the blocking flow from forward edges and adding it to backward edges.
- Finally, the maximum flow is obtained by summing up the flows on all outgoing edges from the source.
Dinic's algorithm is efficient due to the concept of layering. The algorithm increases the flow in the network by finding multiple disjoint augmenting paths in each iteration, resulting in faster convergence compared to the Ford-Fulkerson algorithm.
The time complexity of Dinic's algorithm is O(V^2E), where V is the number of vertices and E is the number of edges. However, with the use of advanced data structures like dynamic trees, the time complexity can be improved to O(V^2E log(V)).
Overall, Dinic's algorithm is a powerful method for solving the maximum flow problem, particularly for large-scale networks.
Here's an example code snippet to run the Dinic's Algorithm:
|nodes edges dinic value|
nodes := #( $1 $2 $3 $4 $5 $6 ).
edges := #( #( $1 $2 10 ) #( $1 $3 10 )
#( $2 $3 2 ) #( $2 $4 4 ) #( $2 $5 8 )
#( $3 $5 9 ) #( $4 $6 10 )
#( $5 $4 6 ) #( $5 $6 10 )).
dinic := AIDinic new.
dinic nodes: nodes.
dinic
edges: edges
from: [ :each | each first ]
to: [ :each | each second ]
capacity: [ :each | each third ].
dinic setStartNode: (dinic findNode: $1) sinkNode: (dinic findNode: $6).
value:= dinic run.
value.
"
Class {
#name : #AIDinic,
#superclass : #AIGraphAlgorithm,
#instVars : [
'adjList',
'queue',
'start',
'sink'
],
#category : #'AI-Algorithms-Graph'
}

{ #category : #configuration }
AIDinic >> addAdjList [

| arr fromNode |
self nodes do: [ :n | adjList at: n put: OrderedCollection new ].
self edges doWithIndex: [ :e :i |
fromNode := e from.
arr := adjList at: fromNode.
arr add: i ]
]

{ #category : #accessing }
AIDinic >> adjList [

^ adjList
]

{ #category : #backtracking }
AIDinic >> bfs [
"This method uses bfs on the residual graph to construct a level graph.
The level graph assigns levels or distances to each node, indicating the shortest path from the source.
The returnBool boolean indicates if there exists and augmenting path(path from source to sink) in the residual level graph."

| node ind returnBool |
[ queue isNotEmpty ] whileTrue: [
node := queue removeFirst.
ind := adjList at: node.
ind do: [ :i |
| e n |
e := edges at: i.
n := e to.
e capacity - e flow >= 1 & (n level = -1) ifTrue: [
n level: node level + 1.
queue addLast: n ] ] ].
returnBool := sink level == -1.
^ returnBool
]

{ #category : #configuration }
AIDinic >> currentIndexSetup [

self nodes do: [ :n | n currentIndex: 1 ]
]

{ #category : #backtracking }
AIDinic >> dfs: fromNode pushed: p [
"This method uses DFS-style algorithm to find a blocking flow (A blocking flow is a flow that saturates all the edges on the path, preventing any further flow) along the augmenting path.
This performs a series of depth-first searches on the residual graph, exploring the edges with positive residual capacity."

| arr cid edg reverseEdg toNode tr min |
p = 0 ifTrue: [ ^ 0 ].
fromNode == sink ifTrue: [ ^ p ].

arr := adjList at: fromNode.
cid := fromNode currentIndex.
tr := 0.

[ cid <= arr size ] whileTrue: [
edg := edges at: (arr at: cid).
toNode := edg to.
"Checking if the TO NODE is one level greater than FROM NODE, and for positive residual capacity"
fromNode level + 1 = toNode level & (edg capacity - edg flow >= 1)
ifTrue: [
fromNode currentIndex: cid.
"Calculating the minimum value between the current positive residual capacity and the pushed value from previous iterations"
min := self minimumValue: edg capacity - edg flow compare: p.
"The recursive call of DFS"
tr := self dfs: toNode pushed: min ].

tr = 0 ifFalse: [
edg flow: edg flow + tr.
reverseEdg := self reverseEdge: toNode to: fromNode.
reverseEdg flow: reverseEdg flow - tr.
^ tr ].
cid := cid + 1.
fromNode currentIndex: cid ].
^ 0
]

{ #category : #configuration }
AIDinic >> edgeClass [

^ AINetworkFlowEdge
]

{ #category : #'building - graph' }
AIDinic >> edges: aCollection from: source to: target capacity: capacityFunction [

| edge edgeRev|
aCollection do: [ :eModel |
edge := self addEdge: eModel from: source to: target.
edge ifNotNil: [ edge capacity: (capacityFunction value: eModel) ].
edgeRev := self addEdge: eModel from: target to: source.
edgeRev ifNotNil: [ edgeRev capacity:0 ].]
]

{ #category : #initialization }
AIDinic >> initialize [

super initialize.
adjList := Dictionary new.
queue := LinkedList new
]

{ #category : #configuration }
AIDinic >> initializeBfs [
"This method
- Initialises all the levels and current index of the nodes
- Runs the bfs method to get the level graph
- Returns a boolean based on whether the there exists an augmenting path or not."

self levelSetup.
self currentIndexSetup.
start level: 0.
queue addLast: start.
^ self bfs
]

{ #category : #configuration }
AIDinic >> levelSetup [

self nodes do: [ :n | n level: -1 ]
]

{ #category : #utilities }
AIDinic >> minimumValue: firstNumber compare: secondNumber [

firstNumber > secondNumber ifTrue: [ ^ secondNumber ].
^ firstNumber
]

{ #category : #configuration }
AIDinic >> nodeClass [

^ AIDinicNode
]

{ #category : #utilities }
AIDinic >> reverseEdge: fromNode to: toNode [

| e |
(adjList at: fromNode) do: [ :i |
e := edges at: i.
e to == toNode & (e capacity = 0) ifTrue: [ ^ e ] ]
]

{ #category : #running }
AIDinic >> run [

| finalFlow pushed breakLoop |
self addAdjList.
finalFlow := 0.
breakLoop := true.

self initializeBfs ifTrue: [ ^ 0 ].

[ breakLoop ] whileTrue: [
pushed := self dfs: start pushed: SmallInteger maxVal.
[ pushed = 0 ] whileFalse: [
finalFlow := finalFlow + pushed.
pushed := self dfs: start pushed: SmallInteger maxVal ].
self initializeBfs ifTrue: [ breakLoop := false ] ].
^ finalFlow
]

{ #category : #initialization }
AIDinic >> setStartNode: startNode sinkNode: sinkNode [

start := startNode.
sink := sinkNode
]

0 comments on commit 876fd9c

Please sign in to comment.