From 12acb13a71c35a57db136d510edca9f5d7c4c6fe Mon Sep 17 00:00:00 2001 From: Anthony C Date: Wed, 23 Aug 2017 21:06:26 -0400 Subject: [PATCH] Kotlin - Forth complete --- .../forth/src/main/kotlin/ForthEvaluator.kt | 127 ++++++++++++++++-- 1 file changed, 114 insertions(+), 13 deletions(-) diff --git a/kotlin/forth/src/main/kotlin/ForthEvaluator.kt b/kotlin/forth/src/main/kotlin/ForthEvaluator.kt index 6979fd3..724529f 100644 --- a/kotlin/forth/src/main/kotlin/ForthEvaluator.kt +++ b/kotlin/forth/src/main/kotlin/ForthEvaluator.kt @@ -1,24 +1,117 @@ +import sun.invoke.empty.Empty import java.util.* class ForthEvaluator { - val stack: Stack = Stack() - val overridenOperators = mutableMapOf() + val stringStack: Stack = Stack() + val intStack: Stack = Stack() + + val overridenOperators = mutableMapOf() private val validOps = listOf("+", "-", "*", "/", "dup", "drop", "swap", "over") - fun evaluateProgram(inpList: List): List{ - for (i in inpList){ - val tokenized = i.split(' ') - tokenized.all { validateToken(it) } - for (j in tokenized){ - stack.push(j) + fun evaluateProgram(inpList: List): List { + for (i in inpList) { + val rawTokenized: List = i.split(' ') + if (":" == rawTokenized.first()) { + overridenOperators[rawTokenized[1]] = rawTokenized.subList(2, rawTokenized.size - 1).joinToString(separator = " ") + try { + if (overridenOperators[rawTokenized[1]]!!.toInt() is Int){ + throw IllegalArgumentException("Cannot redefine numbers") + } + } catch (e: NumberFormatException){} + } else { + val tokenized = doReplacements(rawTokenized).split(' ') + tokenized.all { validateToken(it) } + for (j in tokenized) { + when (j.toLowerCase()) { + "+" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(x + y) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Addition requires that the stack contain at least 2 values") + } + } + "-" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(y - x) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Subtraction requires that the stack contain at least 2 values") + } + } + "*" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(y * x) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Multiplication requires that the stack contain at least 2 values") + } + } + "/" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(y / x) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Division requires that the stack contain at least 2 values") + } catch (e: ArithmeticException) { + throw IllegalArgumentException("Division by 0 is not allowed") + } + } + "dup" -> { + try { + val x = intStack.pop() + intStack.push(x) + intStack.push(x) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Duplicating requires that the stack contain at least 1 value") + } + } + "drop" -> { + try { + intStack.pop() + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Dropping requires that the stack contain at least 1 value") + } + } + "swap" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(x) + intStack.push(y) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Swapping requires that the stack contain at least 2 values") + } + } + "over" -> { + try { + val x = intStack.pop() + val y = intStack.pop() + intStack.push(y) + intStack.push(x) + intStack.push(y) + } catch (e: EmptyStackException) { + throw IllegalArgumentException("Overing requires that the stack contain at least 2 values") + } + } + else -> { + intStack.push(j.toInt()) + } + } + } } } - return emptyList() + return intStack.toList() } + private fun validateToken(token: String): Boolean { - if (token !in validOps) { + if (token.toLowerCase() !in validOps) { try { token.toInt() } catch (e: NumberFormatException) { @@ -29,8 +122,16 @@ class ForthEvaluator { } return true } -} -fun main(args: Array) { - print(ForthEvaluator().evaluateProgram(listOf("1 2 3 4"))) + private fun doReplacements(inpList: List): String { + val returnList = mutableListOf() + for (i in inpList) { + if (i in overridenOperators.keys) { + returnList.add(overridenOperators[i]!!) + } else { + returnList.add(i) + } + } + return returnList.joinToString(separator = " ") + } } \ No newline at end of file