Open Source Your Knowledge, Become a Contributor
Technology knowledge has to be shared and made accessible for free. Join the movement.
Kotlin
Translation Author: @VizGhar
Kotlin is a modern, statically typed programming language designed for conciseness, safety, and interoperability with Java. Kotlin has become the preferred language for Android app development.
@VizGhar provides a tutorial and examples on his GitHub page, but I want to cover 2 things here that differ from my Python AlgorithmXSolver
. The first is a simple switch from tuple
s to data class
es, an idea I wish I had thought of 2 years ago. The second is a difference in code structure due to a minor difference between Python and Kotlin.
Requirements and Actions
Instead of tuple
s for requirements and actions, @VizGhar uses data class
es, a similar, but more powerful container for data elements. Consider the 4 types of requirements for Sudoku. Each cell must be covered with a number, each number must appear in each row, each number must appear in each column and each number must appear in each box. @VizGhar creates 4 separate data classes, each of which inherits from a generic Requirement
.
In my Python, each of my tuple
s begins with a string that identifies one of the 4 requirement types. In @VizGhar’s Kotlin, the class identifies one of the 4 requirement types and the strings become unnecessary.
Before building your solver subclass, you need to define classes for your requirements and actions. In the following code snippet, I have defined the 4 types of requirements and the 1 action type found in Sudoku.
sealed interface Requirement {
data class CellCovered(val row: Int, val col: Int) : Requirement
data class BoxCovered(val box: Int, val value: Char) : Requirement
data class RowCovered(val row: Int, val value: Char) : Requirement
data class ColumnCovered(val col: Int, val value: Char) : Requirement
}
data class Action(val row: Int, val col: Int, val value: Char)
Solver Construction and Initialization
Using my Python solver, the following process has been suggested as standard practice:
- Create a subclass that inherits from
AlgorithmXSolver
. - Override the constructor to build requirements and actions.
- Prior to exiting the subclass constructor, call the superclass constructor passing the newly built requirements and actions.
In Kotlin, your solver subclass cannot override the inherited constructor to build requirements and actions before passing them to the inherited constructor. Requirements and actions must be built before initializing your subclass solver. In the following 9x9 Sudoku example, I have added a function called createSolver
(per @VizGhar's instructions), which builds the requirements and actions before creating an instance of SudokuSolver
.
Example - 9x9 Sudoku
class SudokuSolver(requirements: List<Requirement>, actions: Map<Action, List<Requirement>>) :
DLXSolver<Requirement, Action>(requirements, actions) {
override fun processSolution(solution: List<Action>): Boolean {
// Using the Actions in the solution, build a grid and print the solved Sudoku.
return true // Return true to stop looking for more solutions.
}
}
fun createSolver(grid: List<CharArray>, values: String): SudokuSolver {
val requirements = mutableListOf<Requirement>()
// Build the requirements.
val actions = mutableMapOf<Action, List<Requirement>>()
// Build the actions.
return SudokuSolver(requirements, actions)
}
fun main() {
createSolver(List(9) { readln().toCharArray() }, "123456789").solve()
}
The Solver Code
For even more details, code examples and most importantly, the Kotlin DLXSolver
code, please visit @VizGhar's GitHub repository.