| Safe Haskell | None | 
|---|---|
| Language | Haskell2010 | 
Effectful.Concurrent.MVar.Strict
Description
Lifted Control.Concurrent.MVar.Strict.
Since: 2.4.0.0
Synopsis
- data Concurrent (a :: Type -> Type) b
- runConcurrent :: forall (es :: [Effect]) a. (HasCallStack, IOE :> es) => Eff (Concurrent ': es) a -> Eff es a
- data MVar' a
- newEmptyMVar' :: forall (es :: [Effect]) a. Concurrent :> es => Eff es (MVar' a)
- newMVar' :: forall (es :: [Effect]) a. Concurrent :> es => a -> Eff es (MVar' a)
- takeMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es a
- putMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es ()
- readMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es a
- swapMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es a
- tryTakeMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es (Maybe a)
- tryPutMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es Bool
- tryReadMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es (Maybe a)
- isEmptyMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es Bool
- withMVar' :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es b) -> Eff es b
- withMVar'Masked :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es b) -> Eff es b
- modifyMVar' :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es (a, b)) -> Eff es b
- modifyMVar'_ :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> (a -> Eff es a) -> Eff es ()
- modifyMVar'Masked :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es (a, b)) -> Eff es b
- modifyMVar'Masked_ :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> (a -> Eff es a) -> Eff es ()
- mkWeakMVar' :: forall (es :: [Effect]) a. (HasCallStack, Concurrent :> es) => MVar' a -> Eff es () -> Eff es (Weak (MVar' a))
Effect
data Concurrent (a :: Type -> Type) b Source #
Provide the ability to run Eff computations concurrently in multiple threads and communicate between them.
Warning: unless you stick to high level functions from the withAsync family, the Concurrent effect makes it possible to escape the scope of any scoped effect operation. Consider the following:
>>>import Effectful.Reader.Static qualified as R
>>>printAsk msg = liftIO . putStrLn . (msg ++) . (": " ++) =<< R.ask
>>>:{runEff . R.runReader "GLOBAL" . runConcurrent $ do a <- R.local (const "LOCAL") $ do a <- async $ do printAsk "child (first)" threadDelay 20000 printAsk "child (second)" threadDelay 10000 printAsk "parent (inside)" pure a printAsk "parent (outside)" wait a :} child (first): LOCAL parent (inside): LOCAL parent (outside): GLOBAL child (second): LOCAL
Note that the asynchronous computation doesn't respect the scope of local, i.e. the child thread still behaves like it's inside the local block, even though the parent thread already got out of it.
This is because the value provided by the Reader effect is thread local, i.e. each thread manages its own version of it. For the Reader it is the only reasonable behavior, it wouldn't be very useful if its "read only" value was affected by calls to local from its parent or child threads.
However, the cut isn't so clear if it comes to effects that provide access to a mutable state. That's why statically dispatched State and Writer effects come in two flavors, local and shared:
>>>import Effectful.State.Static.Local qualified as SL>>>:{runEff . SL.execState "Hi" . runConcurrent $ do replicateConcurrently_ 3 $ SL.modify (++ "!") :} "Hi"
>>>import Effectful.State.Static.Shared qualified as SS>>>:{runEff . SS.execState "Hi" . runConcurrent $ do replicateConcurrently_ 3 $ SS.modify (++ "!") :} "Hi!!!"
In the first example state updates made concurrently are not reflected in the parent thread because the value is thread local, but in the second example they are, because the value is shared.
Instances
| type DispatchOf Concurrent Source # | |
| Defined in Effectful.Concurrent.Effect | |
| data StaticRep Concurrent Source # | |
| Defined in Effectful.Concurrent.Effect | |
Handlers
runConcurrent :: forall (es :: [Effect]) a. (HasCallStack, IOE :> es) => Eff (Concurrent ': es) a -> Eff es a Source #
Run the Concurrent effect.
MVar
Strict (WHNF) version of MVar.
Instances
| NFData1 MVar' | |
| Defined in Control.Concurrent.MVar.Strict | |
| NFData (MVar' a) | |
| Defined in Control.Concurrent.MVar.Strict | |
| Eq (MVar' a) | |
newEmptyMVar' :: forall (es :: [Effect]) a. Concurrent :> es => Eff es (MVar' a) Source #
Lifted newEmptyMVar'.
newMVar' :: forall (es :: [Effect]) a. Concurrent :> es => a -> Eff es (MVar' a) Source #
Lifted newMVar'.
takeMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es a Source #
Lifted takeMVar'.
putMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es () Source #
Lifted putMVar'.
readMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es a Source #
Lifted readMVar'.
swapMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es a Source #
Lifted swapMVar'.
tryTakeMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es (Maybe a) Source #
Lifted tryTakeMVar'.
tryPutMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> a -> Eff es Bool Source #
Lifted tryPutMVar'.
tryReadMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es (Maybe a) Source #
Lifted tryReadMVar'.
isEmptyMVar' :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> Eff es Bool Source #
Lifted isEmptyMVar'.
withMVar' :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es b) -> Eff es b Source #
Lifted withMVar'.
withMVar'Masked :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es b) -> Eff es b Source #
Lifted withMVar'Masked.
modifyMVar' :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es (a, b)) -> Eff es b Source #
Lifted modifyMVar'.
modifyMVar'_ :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> (a -> Eff es a) -> Eff es () Source #
Lifted modifyMVar'_.
modifyMVar'Masked :: forall (es :: [Effect]) a b. Concurrent :> es => MVar' a -> (a -> Eff es (a, b)) -> Eff es b Source #
Lifted modifyMVar'Masked.
modifyMVar'Masked_ :: forall (es :: [Effect]) a. Concurrent :> es => MVar' a -> (a -> Eff es a) -> Eff es () Source #
Lifted modifyMVar'Masked_.
mkWeakMVar' :: forall (es :: [Effect]) a. (HasCallStack, Concurrent :> es) => MVar' a -> Eff es () -> Eff es (Weak (MVar' a)) Source #
Lifted mkWeakMVar'.
Note: the finalizer will run a cloned environment, so any changes it makes to thread local data will not be visible outside of it.