差分约束系统 (System of Difference Constraints),是求解关于一组变数的特殊不等式组 之方法。
如果一个系统由
n
{\displaystyle n}
个变量和
m
{\displaystyle m}
个约束条件组成,其中若每个约束条件形如
x
j
−
x
i
≤
b
k
(
i
,
j
∈
[
1
,
n
]
,
k
∈
[
1
,
m
]
)
{\displaystyle x_{j}-x_{i}\leq b_{k}(i,j\in [1,n],k\in [1,m])}
,则称其为差分约束系统。亦即,差分约束系统是求解关于一组变量的特殊不等式组的方法。
求解差分约束系统,可以转化成求解图论 的单源最短路径 。观察
x
j
−
x
i
≤
b
k
{\displaystyle x_{j}-x_{i}\leq b_{k}}
,会发现它类似最短路中的三角不等式
d
[
v
]
≤
d
[
u
]
+
w
[
u
,
v
]
{\displaystyle d[v]\leq d[u]+w[u,v]}
,即
d
[
v
]
−
d
[
u
]
≤
w
[
u
,
v
]
{\displaystyle d[v]-d[u]\leq w[u,v]}
。因此,以每个变数
x
i
{\displaystyle x_{i}}
为结点,对于约束条件
x
j
−
x
i
≤
b
k
{\displaystyle x_{j}-x_{i}\leq b_{k}}
,连接一条边
(
i
,
j
)
{\displaystyle (i,j)}
,边权为
b
k
{\displaystyle b_{k}}
。再增加一个原点
(
s
,
s
)
{\displaystyle (s,s)}
与所有定点 相连,边权 均为0(在某些题目中可能需要根据实际情况进行改动)。对这个图以s为原点运行Bellman-Ford 算法 (或SPFA ),最终
{
d
[
i
]
}
{\displaystyle \{d[i]\}}
即为一组可行解。
例如,考虑这样一个问题,寻找一个5维向量
x
=
(
x
i
)
{\displaystyle x=(x_{i})}
以满足:
这一问题等价于找出未知量
x
i
,
i
∈
{
1
,
2
,
3
,
4
,
5
}
{\displaystyle x_{i},i\in \{1,2,3,4,5\}}
,满足下列8个差分约束条件:
x
2
−
x
5
≤
1
{\displaystyle x_{2}-x_{5}\leq 1}
x
1
−
x
2
≤
0
{\displaystyle x_{1}-x_{2}\leq 0}
x
1
−
x
5
≤
−
1
{\displaystyle x_{1}-x_{5}\leq -1}
x
3
−
x
1
≤
5
{\displaystyle x_{3}-x_{1}\leq 5}
x
4
−
x
1
≤
4
{\displaystyle x_{4}-x_{1}\leq 4}
x
4
−
x
3
≤
−
1
{\displaystyle x_{4}-x_{3}\leq -1}
x
5
−
x
3
≤
−
3
{\displaystyle x_{5}-x_{3}\leq -3}
x
5
−
x
4
≤
−
3
{\displaystyle x_{5}-x_{4}\leq -3}
该问题的一个解为
x
=
(
−
5
,
−
3
,
0
,
−
1
,
−
4
)
{\displaystyle x=(-5,-3,0,-1,-4)}
,另一个解
y
=
(
0
,
2
,
5
,
4
,
1
)
{\displaystyle y=(0,2,5,4,1)}
,这2个解是有联系的:
y
{\displaystyle y}
中的每个元素比
x
{\displaystyle x}
中相应的元素大5。
引理 :设
x
=
(
x
1
,
x
2
,
⋯
,
x
n
)
{\displaystyle x=(x_{1},x_{2},\cdots ,x_{n})}
是差分约束系统
A
x
≤
b
{\displaystyle Ax\leq b}
的一个解,d为任意常数,则
x
+
d
=
(
x
1
+
d
,
x
2
+
d
,
⋯
,
x
n
+
d
)
{\displaystyle x+d=(x_{1}+d,x_{2}+d,\cdots ,x_{n}+d)}
也是该系统
A
x
≤
b
{\displaystyle Ax\leq b}
的一个解。
Bellman-Ford 算法 伪代码:
# 初始化
for each v in V do
d[v] ← ∞;
d[source] ← 0
# 松弛
for i =1,...,|V|-1 do
for each edge (u,v) in E do
d[v] ← min{d[v], d[u]+w(u,v)}
# 检查负环
for each edge (u, v) in E do
if d[v]> d[u] + w(u,v) then
<无解>