Sorting w. r. t. a partial ordering
2022-09-03, post № 263
programming, #haskell, #order-theory
Task. Given a type , a finite array with and together with a finite set of pairs , compute a permutation such that respects the strict partial ordering induced by , provided is loop-free.
Furthermore choose to respect relative ordering of equal elements, that is
An application of late is computing the execution order of inter-dependent transformers.
Possibly due to the seeming banality of the application, my many first presumptuous attempts were both of questionable algorithmic complexity as well as failing in subtle ways, fostering bugs which only seldom showed their ugly faces. Fed up with hotfixed bypasses and manual pre-ordering, I badly longed after an implementation which indeed took some time to get right:
Heart of my approach I will herein present is computing a total ordering on all order-participating values that is compatible with and thus an apt choice for a sorting key to attain by arbitrary extension to .
For this consider the digraph and inductively deconstruct it to form a -conforming array . Care must be taken, however, to correctly position elements left of a node of minimal degree three as e. g. greedily seeking chains may disregard local element relations. As such, iteratively pluck an arbitrarily chosen minimal element, laying these out from left to right via . Their minimality ensures that no element can contradict going down.
When using a stable sorting algorithm which sorts under , relative ordering is naturally respected as desired.
I wrote a quadratic implementation in Haskell that only stipulates Eq a
: sort-via.hs. To achieve the complexity typically sought after for sorting tasks, I suspect an underlying total order on the elements is required for data management. Avoiding Haskell’s non-ADT containers and disappointing standard library support for random number generation, I wrote a C++ implementation and test suite testing both: sort-via.cpp, Makefile.