Предположим, что функция зациклена для получения числового результата.Цикл останавливается либо при достижении максимума итераций, либо при выполнении условия «оптимальности».В любом случае значение из токовой петли выводится.Каков функциональный способ получить как этот результат, так и причину остановки?
Для иллюстрации, вот моя реализация Scala примера "Square Roots" в 4.1 из https://www.cs.kent.ac.uk/people/staff/dat/miranda/whyfp90.pdf.
object SquareRootAlg {
def next(a: Double)(x: Double): Double = (x + a/x)/2
def repeat[A](f: A=>A, a: A): Stream[A] = a #:: repeat(f, f(a))
def loopConditional[A](stop: (A, A) => Boolean)(s: => Stream[A] ): A = s match {
case a #:: t if t.isEmpty => a
case a #:: t => if (stop(a, t.head)) t.head else loopConditional(stop)(t)}
}
Например, чтобы найти квадратный корень из 4:
import SquareRootAlg._
val cond = (a: Double, b: Double) => (a-b).abs < 0.01
val alg = loopConditional(cond) _
val s = repeat(next(4.0), 4.0)
alg(s.take(3)) // = 2.05, "maxIters exceeded"
alg(s.take(5)) // = 2.00000009, "optimality reached"
Этот код работает, но не дает мне причины остановки.Поэтому я пытаюсь написать метод
def loopConditionalInfo[A](stop: (A, A)=> Boolean)(s: => Stream[A]): (A, Boolean)
, выводящий (2.05, false)
в первом случае выше и (2.00000009, true)
во втором.Есть ли способ написать этот метод без изменения методов next
и repeat
?Или другой функциональный подход будет работать лучше?