- Notation
- Needleman-Wunsch: where it all begins
- Dijkstra/BFS: visiting fewer states
- Band doubling: Dijkstra, but more efficient
- GapCost: A first heuristic
- Computational volumes: an even smaller search
- Cheating: an oracle gave us \(g^*\)
- A*: Better heuristics
- Broken idea: A* and computational volumes
- Local doubling
- Diagonal Transition
- A* with Diagonal Transition and pruning: doing less work
- Goal: Diagonal Transition + pruning + local doubling
- Pruning: Improving A* heuristics on the go
- Cheating more: an oracle gave us the optimal path
- TODO: aspriation windows
\begin{equation*} \newcommand{\st}[2]{\langle #1,#2\rangle} \newcommand{\g}{g^*} \newcommand{\fm}{f_{max}} \newcommand{\gap}{\operatorname{Gap}} \end{equation*}
This post is an overview/review of existing alignment algorithms, explained using visualizations. This builds up to the introduction of local doubling, an idea I had while writing a previous post on pre-pruning. My hope is that this can combine the efficiency of Needleman-Wunsch or Diagonal Transition implementations (e.g. Edlib and WFA using bitpacking and SIMD), while having the better complexity of A*-based alignment.
See also my review post on pairwise alignment, comparing algorithms and their runtime/memory trade-offs. This post aims to be visual and intuitive, while that one (as for now) is more formal/technical.
Notation
- Input: sequences of length \(n\) and \(m\). For complexity analysis, we assume that \(m=n\).
- States of the alignment graph:
- states \(u = \st ij\),
- start \(s = \st 00\),
- end \(t = \st nm\).
- Distances:
- distance between vertices \(d(u, v)\),
- distance from start \(g = g(u) := d(s, u)\),
- gapcost: \(\gap(u, v) = \gap(\st ij, \st {i’}{j’}) = |(i’-i) - (j’-j)|\).
Needleman-Wunsch: where it all begins
- Invented by (Needleman and Wunsch 1970) and others. See the review post.
- Visits all states \(\st ij\) of the DP table ordered by column \(i\).
- In fact, any topological sort (total order) of the states of the underlying dependency graph (partially ordered set) is allowed. One particular optimization is process anti-diagonals in order. Those correspond to anti-chains in the poset, and hence can be computed independently of each other.
- Complexity: \(O(n^2)\).
data:image/s3,"s3://crabby-images/2f3c0/2f3c03c94a65ce413ed248b62c77a9ec6d7146d8" alt="Figure 1: NW expands all 1000x1000 states. Ignore the right half for now."
Figure 1: NW expands all 1000x1000 states. Ignore the right half for now.
Dijkstra/BFS: visiting fewer states
Mentioned in Ukkonen (1985)
Visit states ordered by the distance from start \(s\): \(g(u) := d(s, u)\).
Visits exactly those states with
\begin{equation} g(u) \leq \g,\label{dijsktra} \end{equation}
where \(\g := d(s, t)\) is the total alignment distance.
Drawback: much slower:
- bithacks/SIMD don’t work,
- queues are slow compared to iteration.
Complexity: \(O(n d)\), where \(d=\g\) is the edit distance.
- Typical notation is \(O(ns)\) for arbitrary scores and \(O(nd)\) for edit distance. I’d use \(O(n\g)\) for consistency but it’s ugly and non-standard.
data:image/s3,"s3://crabby-images/f4cfc/f4cfc6ea8cd2d39c480cf9850872ff481ce8746a" alt="Figure 2: Dijkstra"
Figure 2: Dijkstra
Band doubling: Dijkstra, but more efficient
- Introduced in Ukkonen (1985)
- Idea: use NW’s efficient column-order to compute only the states with \(g(u) \leq \g\).
- We don’t know \(\g\), but we can find it using exponential search or band doubling.
- For a guess \(\fm\) of \(\g\):
Compute all states with
\begin{equation} \gap(s, u) \leq \fm,\label{doubling} \end{equation}
where \(\gap(s, \st ij)\) is the cost of indels (gapcost) needed to go from \(s\) to \(\st ij\). For edit distance this is simply \(|i-j|\).
When \(g(t) \leq \fm\), all states with \(g\leq \g =g(t) \leq \fm\) have been computed, and we found an optimal path.
Otherwise, double the value of \(\fm\) and try again.
- Starting with \(\fm = 1\), this takes \(T:=1n + 2n + \dots + f_{last}\cdot n\) time. Since \(f_{last}/2 < \g\) we have \(T= (2f_{last}-1)n \leq 4\g n\).
- Complexity: \(O(nd)\)
data:image/s3,"s3://crabby-images/d6f9c/d6f9c55ce2031ddeb91ddd08f6ef1a9950d40620" alt="Figure 3: NW + doubling"
Figure 3: NW + doubling
GapCost: A first heuristic
We can sharpen \eqref{doubling} by also bounding the indel cost (gapcost) from \(u\) to the end:
\begin{equation} \gap(s, u)+\gap(u, t) \leq \fm,\label{doubling-gap} \end{equation}
Assuming both input sequences are the same length (\(m=n\)), this halves the runtime.
This can also be used on top of Dijkstra to give a first A* variant where states are ordered by \(f(u) := g(u) + \gap(u, t)\).
It is possible to transform the insertion and deletion costs in a way that already accounts for the gapcost, see this post.
data:image/s3,"s3://crabby-images/03ed0/03ed0de6b747be66b92e1ce74987a4b2f0ebf7d5" alt="Figure 4: NW + doubling + gapcost"
Figure 4: NW + doubling + gapcost
Computational volumes: an even smaller search
Introduced in Spouge (1989)
Equations \eqref{doubling} and \eqref{doubling-gap} determine the area to be computed up-front. But we can make a simple improvement and take into account the current distance \(g(u) \geq \gap(s, u)\):
\begin{equation} g(u)+\gap(u, t) \leq \fm.\label{volume-gap} \end{equation}
An even simpler option is \(g(u) \leq \fm\), which corresponds directly to computing increasing portions of Dijkstra.
This still relies on repeated doubling of \(\fm\).
data:image/s3,"s3://crabby-images/1888c/1888c6fbf7354242a04fd8a0eb1faaa254c68e96" alt="Figure 5: NW + doubling + g + gapcost"
Figure 5: NW + doubling + g + gapcost
Cheating: an oracle gave us \(g^*\)
- If we already know the target distance \(\g\), we can skip the exponential
search over \(\fm\) and directly use \(\fm = \g\). This will speed up all of the
band doubling algorithms above up to \(4\) times:
- no need to try smaller \(\fm<\g\) => \(2x\) faster,
- no more unlucky cases where \(\fm=2\g-\epsilon\).
- More generally, we can make an initial guess for \(\fm\) if we roughly know the distance distribution of the input.
A*: Better heuristics
Instead of visiting states by column \(i\) or distance \(g\), we can order by
\begin{equation} f(u) := g(u)+h(u) \leq \g,\label{astar} \end{equation}
where \(h\) is any heuristic function satisfying \(h(u) \leq d(u, t)\).
Drawback: Again, A* is slow because of the priority queue and many computations of \(h\).
data:image/s3,"s3://crabby-images/63b7d/63b7d50a6eb8a2dcf1759a2bd5c39d9afa224f47" alt="Figure 6: A* + CSH + gapcost"
Figure 6: A* + CSH + gapcost
Broken idea: A* and computational volumes
- Just like band doubling speeds up Dijkstra, can we use it to speed up A*?
- Start with \(\fm = h(s)\).
- Compute all states with \(f(u) \leq \fm\) in column-order.
- Double \(\fm\) after each try.
- BROKEN: If we start with \(\fm = h(s) = \g-1\) and we double to \(\fm = 2\g-2\) the number of expanded states goes from \(O(n)\) to \(O(n^2)\).
data:image/s3,"s3://crabby-images/7a9ed/7a9ed26221f1be64946b0333625b3ae1ffc4dbcf" alt="Figure 7: NW + CSH + gapcost + Doubling"
Figure 7: NW + CSH + gapcost + Doubling
Local doubling
Without heuristic
data:image/s3,"s3://crabby-images/d186a/d186aee5c1b924f769778d6d90f8aac46bd7ed58" alt="Figure 8: NW + Local-Doubling"
Figure 8: NW + Local-Doubling
With heuristic
data:image/s3,"s3://crabby-images/ae464/ae46461be554da9bc3622791834b3173f8de9c6b" alt="Figure 9: NW + CSH + gapcost + Local-Doubling"
Figure 9: NW + CSH + gapcost + Local-Doubling
Diagonal Transition
data:image/s3,"s3://crabby-images/cb63c/cb63c1ec31c46a4b4008eae71c08051153bc39b5" alt="Figure 10: DT"
Figure 10: DT
A* with Diagonal Transition and pruning: doing less work
data:image/s3,"s3://crabby-images/a8cc6/a8cc6d5210415f5915d604f50d53cf14c36fdee8" alt="Figure 11: Astar + DT"
Figure 11: Astar + DT
Goal: Diagonal Transition + pruning + local doubling
data:image/s3,"s3://crabby-images/a2133/a21330c1a465d42c362253fbac6b785de1d3acb7" alt="Figure 12: DT + CSH + gapcost + Local-Doubling"
Figure 12: DT + CSH + gapcost + Local-Doubling
Here’s a gif with pruning as well:
data:image/s3,"s3://crabby-images/9922b/9922b8829ccdc8580e1d0dadcfcea8f3a2053f05" alt="Figure 13: DT + CSH + gapcost + pruning + Local-Doubling"
Figure 13: DT + CSH + gapcost + pruning + Local-Doubling
Pruning: Improving A* heuristics on the go
Cheating more: an oracle gave us the optimal path
- Pruning brings a challenge to the local
TODO: aspriation windows
In chess engines (ie alpha beta search/pruning) there is the concept of aspiration window which is similar to exponential search. Maybe we can reuse some concepts.