GHC отклоняет код монады ST как неспособный объединить переменные типа? - PullRequest
7 голосов
/ 04 ноября 2011

Я написал следующую функцию:

(.>=.) :: Num a => STRef s a -> a -> Bool
r .>=. x = runST $ do
 v <- readSTRef r
 return $ v >= x

но когда я попытался скомпилировать, я получил следующую ошибку:

Could not deduce (s ~ s1)
from the context (Num a)
  bound by the type signature for
             .>=. :: Num a => STRef s a -> a -> Bool
  at test.hs:(27,1)-(29,16)
  `s' is a rigid type variable bound by
      the type signature for .>=. :: Num a => STRef s a -> a -> Bool
      at test.hs:27:1
  `s1' is a rigid type variable bound by
       a type expected by the context: ST s1 Bool at test.hs:27:12
Expected type: STRef s1 a
  Actual type: STRef s a
In the first argument of `readSTRef', namely `r'
In a stmt of a 'do' expression: v <- readSTRef r

Кто-нибудь может помочь?

Ответы [ 2 ]

12 голосов
/ 04 ноября 2011

Это точно так, как задумано.STRef действителен только в одном прогоне runST.И вы пытаетесь вставить внешний STRef в новый прогон runST.Это не верно.Это позволило бы произвольные побочные эффекты в чистом коде.

Итак, то, что вы пытаетесь, невозможно достичь.По замыслу!

7 голосов
/ 04 ноября 2011

Вы должны оставаться в контексте ST:

(.>=.) :: Ord a => STRef s a -> a -> ST s Bool
r .>=. x = do
 v <- readSTRef r
 return $ v >= x

(И, как указывает Хаммар, для использования >= вам нужен класс типов Ord, которого Num нетобеспечить.)

...