- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在模块threads中的Control.Concurrent.Thread.Group
包中,有一个函数forkIO
:
forkIO :: ThreadGroup -> IO α -> IO (ThreadId, IO (Result α))
MonadBaseControl
解除它。这是我的尝试:
fork :: (MonadBase IO m) => TG.ThreadGroup -> m α -> m (ThreadId, m (Result α))
fork tg action = control (\runInBase -> TG.forkIO tg (runInBase action))
Couldn't match type `(ThreadId, IO (Result (StM m α)))'
with `StM m (ThreadId, m (Result α))'
Expected type: IO (StM m (ThreadId, m (Result α)))
Actual type: IO (ThreadId, IO (Result (StM m α)))
In the return type of a call of `TG.forkIO'
In the expression: TG.forkIO tg (runInBase action)
In the first argument of `control', namely
`(\ runInBase -> TG.forkIO tg (runInBase action))'
最佳答案
主要问题是IO a
的forkIO
参数。要在m a
中派生IO
操作,我们需要一种将m a
运行到IO a
的方法。为此,我们可以尝试使类具有runBase :: MonadBase b m => m a -> b a
方法的monad,但是很少有有趣的转换器可以提供。例如,如果考虑使用StateT
变压器,那么它首先可以找到如何使用runStateT
在基本monad中运行某些内容的方法,前提是它首先有机会观察其自身的状态。
runFork :: Monad m => StateT s m a -> StateT s m (m b)
runFork x = do
s <- get
return $ do
(a, s') <- runStateT x s
return a
runForkBase :: MonadBase b m => m a -> m (b a)
,我们将在下面的类型类中使用它。
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE FunctionalDependencies #-}
import Control.Monad.Base
class (MonadBase b m) => MonadRunForkBase b m | m -> b where
runForkBase :: m a -> m (b a)
Fork
一词,以强调未来状态更改通常不会在两个未来之间共享。因此,可以提供
WriterT
的少数有趣的转换器(例如
runBase
)仅提供无趣的
runBase
;它们产生的副作用是无法观察到的。
fork
实例提供的有限形式的降低,我们可以为任何内容编写类似
MonadRunForkBase IO m
的内容。我要从基址到
lift
普通的
forkIO
,而不是从
threads的那个,您可以使用相同的方法。
{-# LANGUAGE FlexibleContexts #-}
import Control.Concurrent
forkInIO :: (MonadRunForkBase IO m) => m () -> m ThreadId
forkInIO action = runForkBase action >>= liftBase . forkIO
MonadRunForkBase
实例提供哪些变压器?”马上,我们可以轻松地为具有
MonadBase
实例的任何基本monad提供它们
import Control.Monad.Trans.Identity
import GHC.Conc.Sync (STM)
instance MonadRunForkBase [] [] where runForkBase = return
instance MonadRunForkBase IO IO where runForkBase = return
instance MonadRunForkBase STM STM where runForkBase = return
instance MonadRunForkBase Maybe Maybe where runForkBase = return
instance MonadRunForkBase Identity Identity where runForkBase = return
import Control.Monad.Trans.Class
class (MonadTrans t) => MonadTransRunFork t where
runFork :: Monad m => t m a -> t m (m a)
runForkBaseDefault :: (Monad (t m), MonadTransRunFork t, MonadRunForkBase b m) =>
t m a -> t m (b a)
runForkBaseDefault = (>>= lift . runForkBase) . runFork
MonadRunForkBase
的
StateT
实例。首先,我们将使用上面的
runFork
来创建一个
MonadTransRunFork
实例
import Control.Monad
import qualified Control.Monad.Trans.State.Lazy as State
instance MonadTransRunFork (State.StateT s) where
runFork x = State.get >>= return . liftM fst . State.runStateT x
MonadRunForkBase
实例。
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE UndecidableInstances #-}
instance (MonadRunForkBase b m) => MonadRunForkBase b (State.StateT s m) where
runForkBase = runForkBaseDefault
RWS
做同样的事情
import qualified Control.Monad.Trans.RWS.Lazy as RWS
instance (Monoid w) => MonadTransRunFork (RWS.RWST r w s) where
runFork x = do
r <- RWS.ask
s <- RWS.get
return $ do
(a, s', w') <- RWS.runRWST x r s
return a
instance (MonadRunForkBase b m, Monoid w) => MonadRunForkBase b (RWS.RWST r w s m) where
runForkBase = runForkBaseDefault
MonadRunForkBase
不同,
monad-control中的
MonadBaseControl
并没有基于“未来状态通常不会在两个期货之间共享”的假设。
MonadBaseContol
和
control
努力通过使用
restoreM :: StM m a -> m a
从控制结构中的分支还原状态。对于base的
forkIO
来说,这并不存在问题;使用
forkIO
是
MonadBaseControl
文档中提供的示例。对于
forkIO
from threads,这将是一个小问题,因为返回了额外的
m (Result a)
。
m (Result a)
实际上将作为
IO (Result (StM m a))
返回。我们可以摆脱
IO
并用
m
替换为
liftBase
,而剩下
m (Result (StM m a))
。我们可以将
StM m a
转换成
m a
来恢复状态,然后用
a
返回
restoreM
,但是它被卡在
Result ~ Either SomeException
中。
Either l
是函子,因此我们可以在其中的任何地方应用
restoreM
,将类型简化为
m (Result (m a))
。
Either l
也是
Traversable
,对于任何
Traversable
t
,我们始终可以将其替换为
Monad
或
Applicative
中的
sequenceA :: t (f a) -> f (t a)
。在这种情况下,我们可以使用特殊用途的
mapM
,它是
fmap
和
sequenceA
的组合,只有
Monad
约束。这将给出
m (m (Result a))
,而
m
将通过在Monad中的联接或仅使用
>>=
来展平。这引起了
{-# LANGUAGE FlexibleContexts #-}
import Control.Concurrent
import Control.Concurrent.Thread
import qualified Control.Concurrent.Thread.Group as TG
import Control.Monad.Base
import Control.Monad.Trans.Control
import Data.Functor
import Data.Traversable
import Prelude hiding (mapM)
fork :: (MonadBaseControl IO m) =>
TG.ThreadGroup -> m a -> m (ThreadId, m (Result a))
fork tg action = do
(tid, r) <- liftBaseWith (\runInBase -> TG.forkIO tg (runInBase action))
return (tid, liftBase r >>= mapM restoreM)
m (Result a)
时,它将把状态从分支线程复制到原始线程,这可能会很有用。如果要在读取
Result
之后恢复主线程的状态,则需要首先捕获它。
checkpoint
将捕获整个状态,并返回一个操作来还原它。
checkpoint :: MonadBaseControl b m => m (m ())
checkpoint = liftBaseWith (\runInBase -> runInBase (return ()))
>>= return . restoreM
fork
发生时获取状态,而无需考虑修改另一个线程中的状态。当我们等待主线程中的结果时,主线程中的状态将设置为来自派生线程的状态。我们可以通过运行
checkpoint
创建的操作来获取主线程的状态。
import Control.Monad.State hiding (mapM)
example :: (MonadState String m, MonadBase IO m, MonadBaseControl IO m) => m ()
example = do
get >>= liftBase . putStrLn
tg <- liftBase TG.new
(_, getResult) <- fork tg (get >>= put . ("In Fork:" ++) >> return 7)
get >>= put . ("In Main:" ++)
revert <- checkpoint
result <- getResult
(liftBase . print) result
get >>= liftBase . putStrLn
revert
get >>= liftBase . putStrLn
main = do
runStateT example "Initial"
return ()
Initial
Right 7
In Fork:Initial
In Main:Initial
关于haskell - MonadBaseControl:如何解除ThreadGroup,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27704615/
如何确定操作表何时已关闭或已被驳回? 我关注了这个主题Fitting a UIDatePicker into a UIActionSheet并向操作表添加了选择器 View 。但是,当操作表仍处于打开
在关闭 UIActivityViewController 后如何调用操作?例如,如果我想在用户关闭 UIActivityViewController 后更改一段文本。 谢谢大家! 最佳答案 使用它的完
我的代码中有以下几行: func searchBarTextDidBeginEditing(searchBar: UISearchBar) { searchBar.showsCancelBut
像这样使用 UIAlertController 时: var alert = UIAlertController(title: "Core Location", message: "Loca
我正在使用 UITableViewController 来显示要编辑的项目列表。点击一行后,用户将转到 View Controller 以编辑数据。 当他们通过默认 <(后退)按钮退出编辑屏幕时,我想
我有一个自定义的 UIImagePickerController ,它工作得很好,只是我面临一个我认为应该相当简单的问题 - 我只是还没有找到解决方案。 触摸我自定义添加的“照片”按钮后,我将其定位到
我在显示 UIMenuController 的表格 View 单元格上实现了一个长按手势识别器,当菜单显示时,相应的表格 View 单元格被选中,这是我的要求。但问题是当我触及 UIMenuContr
我有: df = pd.DataFrame({'col1': ['asdf', 'xy', 'q'], 'col2': [1, 2, 3]}) col1 col2 0 asdf 1
我正在尝试解除实时事件的绑定(bind)(已尝试使用 .live 和 .delegate)。 当我想解除绑定(bind)时,我有 3 个不同的选项:解除绑定(bind) "click"、解除绑定(bi
我正在使用 spock、geb 和 WebDriver 编写测试脚本。该脚本在不安全的页面上提交表单。该页面提交到安全的 HTTPS URL。 Firefox 对此显示警告,特别是: 这会导致以下错误
我有两个 s 和每个 ng-repeat,但都对其子元素执行相同的操作,如下所示: {{item.name}} {{item.name}} 想象一下,有两种类型的数据集,要渲染
我构建了一个通过 APIHiJack 连接到 Win32 TextOut 函数的应用程序。当应用程序启动时,DLL 将按预期注入(inject),并且我的新 TextOut 函数被成功调用。 目前,关
我遇到了一次非常奇怪的崩溃,这是回溯。 * thread #1: tid = 0x2403, 0x3379516c CoreFoundation`CFHash + 8, stop reason = E
我有一个 FirstViewController 和一个 SecondViewController。它们的 UINavigationBar 有不同的颜色。当我显示 SecondViewControll
抱歉,如果我在文档中遗漏了一些内容,但无论如何我都找不到阻止在 SweetAlert 2 中关闭对话框的方法,这些将不起作用: await Swal.fire({
有没有人见过这个?在 iPad 模拟器中,我有一个 About View Controller 。我想以模态方式呈现它,并让用户单击关闭按钮。 更复杂的是,我有一个主视图 Controller ,它显
我有 3 个 View Controller - BaseViewController->AviewController->BviewController。 AviewController 以模态方式
如果我在对象上有一个事件聚合器, eventAggregator: _.extend({}, Backbone.Events), 对于模态视图,我基本上让模态视图的呈现者监听模态视图的事件。 this
我有几个 View Controller 。如果 Alert View 被确认,我需要返回到第一个 View Controller。如果没有 unwind Segue,我会这样做: @IBAction
我有一个为 SharePoint 编写的 Accordion 菜单: parent.click(function(event) { event.preventDefault(); va
我是一名优秀的程序员,十分优秀!