-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcirculation.go
175 lines (159 loc) · 5.74 KB
/
circulation.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
// Package flownet provides algorithms for solving maximum-flow and circulation with node and/or edge demands.
// It implements a push-relabel algorithm for maximum flow and provides a wrapper for specifying circulation
// problems.
package flownet
import (
"fmt"
"math"
)
// A Circulation is a flow network which has an additional demand associated with each of its nodes
// or edges. Flow may be supplied to the network via negative node demands.
//
// Whereas in a traditional flow network problem we are interested in maximizing the amount of flow
// from the source to the sink, in a circulation we ask if there is a feasible flow which satisfies
// the demand. Nodes in a circulation are not connected the source or sink as in a traditional flow
// network. Trying to add these connections to a Circulation will cause an error.
type Circulation struct {
FlowNetwork
// demand stores the demand for each edge
demand map[edge]int64
// nodeDemand stores the demand for each node
nodeDemand map[int]int64
// special source node used for node demands
nodeSource int
// special sink node used for node demands
nodeSink int
// amount of flow expected in a valid circulation.
targetValue int64
}
// NewCirculation constructs a new graph allocating initial capacity for the provided number of nodes.
func NewCirculation(numNodes int) Circulation {
return Circulation{
FlowNetwork: NewFlowNetwork(numNodes),
// demand maps from edges (using external nodeIDs) to the demand along each edge.
demand: make(map[edge]int64),
// nodeDemand maps from external nodeIDs to the demand for each node.
nodeDemand: make(map[int]int64),
}
}
// SetNodeDemand sets the demand for a node.
func (c *Circulation) SetNodeDemand(nodeID int, demand int64) error {
if nodeID == Source || nodeID == Sink {
return fmt.Errorf("no demand can be set for the source or sink")
}
if nodeID < 0 || nodeID >= c.numNodes {
return fmt.Errorf("no node with id %d is known", nodeID)
}
if demand != 0 && c.nodeSource == 0 {
c.nodeSource = c.AddNode()
c.nodeSink = c.AddNode()
c.FlowNetwork.AddEdge(c.nodeSink, c.nodeSource, math.MaxInt64)
}
if demand == 0 {
c.FlowNetwork.AddEdge(c.nodeSource, nodeID, 0)
c.FlowNetwork.AddEdge(nodeID, c.nodeSink, 0)
}
if demand > 0 {
c.FlowNetwork.AddEdge(nodeID, c.nodeSink, demand)
}
if demand < 0 {
c.FlowNetwork.AddEdge(c.nodeSource, nodeID, -demand)
}
c.nodeDemand[nodeID] = demand
return nil
}
// AddEdge sets the capacity and non-negative demand of the edge in the circulation. An error is returned
// if either fromID or toID are not valid node IDs. Adding an edge twice has no additional effect.
// Setting demands on edges also updates the demand on the adjacent nodes.
func (c *Circulation) AddEdge(fromID, toID int, capacity, demand int64) error {
if fromID == Source || fromID == Sink || toID == Source || toID == Sink {
// TODO: could source/sink be interpreted as the 'special' nodeSource / nodeSink?
return fmt.Errorf("edges to/from the source/sink nodes cannot be used in a Circulation")
}
if demand < 0 {
return fmt.Errorf("edge demands must be non-zero")
}
if capacity < demand {
return fmt.Errorf("capacity cannot be smaller than demand; capacity = %d, demand = %d", capacity, demand)
}
if err := c.FlowNetwork.AddEdge(fromID, toID, capacity-demand); err != nil {
return err
}
e := edge{fromID, toID}
if demand != 0 {
c.demand[e] = demand
}
if demand == 0 {
delete(c.demand, e)
}
return nil
}
// Underflow is the amount of demand that is unable to be satisfied with the current constraints.
// It is available after running PushRelabel.
func (c *Circulation) Underflow() int64 {
return c.targetValue - c.Outflow()
}
// Capacity returns the capacity of the provided edge.
func (c *Circulation) Capacity(from, to int) int64 {
return c.FlowNetwork.Capacity(from, to) + c.demand[edge{from, to}]
}
// Flow returns the flow achieved by the circulation along the provided edge. The results are
// only meaningful after PushRelabel has been run.
func (c *Circulation) Flow(from, to int) int64 {
return c.FlowNetwork.Flow(from, to) + c.demand[edge{from, to}]
}
// EdgeDemand returns the demand required along each edge.
func (c *Circulation) EdgeDemand(from, to int) int64 {
return c.demand[edge{from, to}]
}
// NodeDemand returns the demand required at each node.
func (c *Circulation) NodeDemand(nodeID int) int64 {
return c.nodeDemand[nodeID]
}
// SatisfiesDemand is true iff the flow satisfies all of the required node and edge demands.
func (c *Circulation) SatisfiesDemand() bool {
return c.Outflow() == c.targetValue
}
// PushRelabel finds a valid circulation (if one exists) via the push-relabel algorithm.
func (c *Circulation) PushRelabel() {
if len(c.demand) == 0 && len(c.nodeDemand) == 0 {
c.FlowNetwork.PushRelabel()
return
}
// disconnect the source and sink nodes; they don't work the same for circulations with demands
for edge := range c.FlowNetwork.capacity {
if edge.from == sourceID {
delete(c.FlowNetwork.capacity, edge)
}
if edge.to == sinkID {
delete(c.FlowNetwork.capacity, edge)
}
}
targetValue := int64(0)
for e, demand := range c.demand {
if demand != 0 {
c.addEdge(e.from, Sink, c.Capacity(e.from, Sink)+demand)
c.addEdge(Source, e.to, c.Capacity(Source, e.to)+demand)
}
if demand > 0 {
targetValue += demand
}
}
for u, demand := range c.nodeDemand {
if demand > 0 {
c.addEdge(u, Sink, demand)
targetValue += demand
}
if demand < 0 {
c.addEdge(Source, u, -demand)
}
}
if targetValue == 0 { // handle no node or edge demands
c.addEdge(Source, c.nodeSource, math.MaxInt64)
c.addEdge(c.nodeSink, Sink, math.MaxInt64)
c.addEdge(c.nodeSink, c.nodeSource, 0)
}
c.targetValue = targetValue
// find the max-flow in the resulting flow network.
c.FlowNetwork.PushRelabel()
}