Полагаю, scalaz 7.0.x и следующие операции импорта (посмотрите историю ответов для scalaz 6.x ):
import scalaz._
import Scalaz._
Тип состояния определяется как State[S, A]
, где S
- это тип состояния, а A
- это тип декорируемого значения. Основной синтаксис для создания значения состояния использует функцию State[S, A]
:
// Create a state computation incrementing the state and returning the "str" value
val s = State[Int, String](i => (i + 1, "str"))
Чтобы выполнить вычисление состояния для начального значения:
// start with state of 1, pass it to s
s.eval(1)
// returns result value "str"
// same but only retrieve the state
s.exec(1)
// 2
// get both state and value
s(1) // or s.run(1)
// (2, "str")
Состояние может быть пропущено через вызовы функций. Чтобы сделать это вместо Function[A, B]
, определите Function[A, State[S, B]]]
. Используйте функцию State
...
import java.util.Random
def dice() = State[Random, Int](r => (r, r.nextInt(6) + 1))
Тогда синтаксис for/yield
может использоваться для составления функций:
def TwoDice() = for {
r1 <- dice()
r2 <- dice()
} yield (r1, r2)
// start with a known seed
TwoDice().eval(new Random(1L))
// resulting value is (Int, Int) = (4,5)
Вот еще один пример. Заполните список с помощью TwoDice()
вычислений состояния.
val list = List.fill(10)(TwoDice())
// List[scalaz.IndexedStateT[scalaz.Id.Id,Random,Random,(Int, Int)]]
Используйте последовательность, чтобы получить State[Random, List[(Int,Int)]]
. Мы можем предоставить псевдоним типа.
type StateRandom[x] = State[Random,x]
val list2 = list.sequence[StateRandom, (Int,Int)]
// list2: StateRandom[List[(Int, Int)]] = ...
// run this computation starting with state new Random(1L)
val tenDoubleThrows2 = list2.eval(new Random(1L))
// tenDoubleThrows2 : scalaz.Id.Id[List[(Int, Int)]] =
// List((4,5), (2,4), (3,5), (3,5), (5,5), (2,2), (2,4), (1,5), (3,1), (1,6))
Или мы можем использовать sequenceU
, который выведет типы:
val list3 = list.sequenceU
val tenDoubleThrows3 = list3.eval(new Random(1L))
// tenDoubleThrows3 : scalaz.Id.Id[List[(Int, Int)]] =
// List((4,5), (2,4), (3,5), (3,5), (5,5), (2,2), (2,4), (1,5), (3,1), (1,6))
Еще один пример с State[Map[Int, Int], Int]
для вычисления частоты сумм в списке выше. freqSum
вычисляет сумму бросков и подсчитывает частоты.
def freqSum(dice: (Int, Int)) = State[Map[Int,Int], Int]{ freq =>
val s = dice._1 + dice._2
val tuple = s -> (freq.getOrElse(s, 0) + 1)
(freq + tuple, s)
}
Теперь используйте траверс, чтобы применить freqSum
к tenDoubleThrows
. traverse
эквивалентно map(freqSum).sequence
.
type StateFreq[x] = State[Map[Int,Int],x]
// only get the state
tenDoubleThrows2.copoint.traverse[StateFreq, Int](freqSum).exec(Map[Int,Int]())
// Map(10 -> 1, 6 -> 3, 9 -> 1, 7 -> 1, 8 -> 2, 4 -> 2) : scalaz.Id.Id[Map[Int,Int]]
Или более кратко, используя traverseU
для вывода типов:
tenDoubleThrows2.copoint.traverseU(freqSum).exec(Map[Int,Int]())
// Map(10 -> 1, 6 -> 3, 9 -> 1, 7 -> 1, 8 -> 2, 4 -> 2) : scalaz.Id.Id[Map[Int,Int]]
Обратите внимание, что поскольку State[S, A]
является псевдонимом типа для StateT[Id, S, A]
, tenDoubleThrows2 в конечном итоге будет напечатан как Id
. Я использую copoint
, чтобы превратить его в тип List
.
Короче говоря, похоже, что ключом к использованию состояния является наличие функций, возвращающих функцию, изменяющую состояние и требуемое фактическое значение результата ... Отказ от ответственности: я никогда не использовал state
в рабочем коде, просто пытался чтобы почувствовать это.
Дополнительная информация о комментарии @ziggystar
Я перестал пытаться использовать stateT
, может быть, кто-то еще может показать, можно ли увеличить StateFreq
или StateRandom
для выполнения комбинированных вычислений. Вместо этого я обнаружил, что состав двух преобразователей состояния можно объединить так:
def stateBicompose[S, T, A, B](
f: State[S, A],
g: (A) => State[T, B]) = State[(S,T), B]{ case (s, t) =>
val (newS, a) = f(s)
val (newT, b) = g(a) apply t
(newS, newT) -> b
}
Он основан на g
, являющемся однопараметрической функцией, принимающей результат первого преобразователя состояния и возвращающей преобразователь состояния. Тогда будет работать следующее:
def diceAndFreqSum = stateBicompose(TwoDice, freqSum)
type St2[x] = State[(Random, Map[Int,Int]), x]
List.fill(10)(diceAndFreqSum).sequence[St2, Int].exec((new Random(1L), Map[Int,Int]()))