[Haskell-cafe] Why can `env` be assigned value two times ?

Ross Mellgren rmm-haskell at z.odi.ac
Sun Nov 15 02:14:38 EST 2009


They're two different 'env's, which can be observed by desugaring the do-notation:

do env <- liftIO (readIORef envRef)
   env <- return (filter (\(_id, _) -> _id /= id) env)
...

Desugaring do-notation gets us:

liftIO (readIORef envRev) >>= \ env ->
return (filter (\(_id, _) -> _id /= id) env) >>= \ env ->
...

Sometimes people use different names to make this obvious, e.g.

do env <- liftIO $ readIORef envRef
   env' <- return (filter ... env)

Also note that you're doing a pure operation here, so you don't need two bindings. You could instead do:

do env <- filter (\(_id, _) -> _id /= id) <$> readIORev envRef
...

(<$> is from the supremely useful Control.Applicative, and is equivalent to fmap from Functor, or liftM from Monad)

or:

do env <- liftIO $ readIORef envRef
   let env' = filter ... env

Using let notation here makes it somewhat more obvious that that line doesn't have any side effects.

-Ross

On Nov 15, 2009, at 2:05 AM, zaxis wrote:

> 
> defineVar :: Env -> (Id, Val) -> IOThrowsError Val
> defineVar envRef (id, val) = do {
>    env <- liftIO $ readIORef envRef;
>    env <- return $ filter (\(_id, _) -> _id/=id) env; -- clear the current
> scope
>    valRef <- liftIO $ newIORef val;
>    liftIO $ writeIORef envRef $ ((id, valRef):env);
>    return val;
> }
> 
> In haskell, the variable canot change its value , right? If so, why can the
> `env` be assigned value twice?
> 
> Sincerely!
> 
> -----
> fac n = foldr (*) 1 [1..n]
> -- 
> View this message in context: http://old.nabble.com/Why-can-%60env%60-be-assigned-value-two-times---tp26356073p26356073.html
> Sent from the Haskell - Haskell-Cafe mailing list archive at Nabble.com.
> 
> _______________________________________________
> Haskell-Cafe mailing list
> Haskell-Cafe at haskell.org
> http://www.haskell.org/mailman/listinfo/haskell-cafe



More information about the Haskell-Cafe mailing list