Implement permutation
This commit is contained in:
parent
f5e38842c2
commit
ade18b5ccf
50
src/dev/w1zzrd/math/Permutation.kt
Normal file
50
src/dev/w1zzrd/math/Permutation.kt
Normal file
@ -0,0 +1,50 @@
|
||||
package dev.w1zzrd.math
|
||||
|
||||
import kotlin.collections.ArrayList
|
||||
|
||||
class Permutation {
|
||||
private val orderedElements: CharArray
|
||||
private val perms: List<CharArray>
|
||||
|
||||
constructor(vararg permutations: CharArray) {
|
||||
orderedElements = permutations.map(CharArray::toSet).reduceRight{ it, acc -> acc.union(it) }.toCharArray()
|
||||
perms = permutations.reversed()
|
||||
}
|
||||
|
||||
constructor(transpositions: ArrayList<Pair<Char, Char>>) {
|
||||
orderedElements = transpositions.map(Pair<Char, *>::first).union(transpositions.map(Pair<*, Char>::second)).toCharArray()
|
||||
perms = transpositions.map { charArrayOf(it.first, it.second) }
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Combine two permutations
|
||||
*/
|
||||
operator fun times(perm: Permutation)
|
||||
= Permutation(this(*perm(*orderedElements)).zip(orderedElements) // Generate transpositions
|
||||
.foldRight(ArrayList<Pair<Char, Char>>()){ // Deduplicate transpositions
|
||||
insert, acc ->
|
||||
acc.firstOrNull {
|
||||
(it.first == insert.first && it.second == insert.second) || (it.first == insert.second && it.second == insert.first)
|
||||
} ?: acc.add(insert)
|
||||
acc
|
||||
})
|
||||
|
||||
operator fun times(perm: CharArray) = this * Permutation(perm)
|
||||
|
||||
operator fun invoke(vararg elements: Char): CharArray {
|
||||
// Apply all permutations
|
||||
for(perm in perms)
|
||||
// Check each element-permutation
|
||||
for(itemIndex in elements.indices)
|
||||
// Check if permutation matches
|
||||
for(index in perm.indices)
|
||||
// If permutation matches, apply it and go to next element
|
||||
if(elements[itemIndex] == perm[index]) {
|
||||
elements[itemIndex] = perm[(index + 1) % perm.size]
|
||||
break
|
||||
}
|
||||
|
||||
return elements
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user