- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我担心 IO 的引入会对程序产生多大影响。假设我的程序深处的一个函数被更改为包含一些 IO;如何隔离此更改,而不必同时更改 IO 路径中的每个函数?
例如,在一个简化的示例中:
a :: String -> String
a s = (b s) ++ "!"
b :: String -> String
b s = '!':(fetch s)
fetch :: String -> String
fetch s = reverse s
main = putStrLn $ a "hello"
(这里的 fetch 更实际的是从静态 Map 中读取一个值作为其结果)但是,假设由于某些业务逻辑发生变化,我需要在某个数据库中查找 fetch 返回的值(我可以在此处通过调用 getLine 来举例说明):
fetch :: String -> IO String
fetch s = do
x <- getLine
return $ s ++ x
所以我的问题是,如何避免重写此链中的每个函数调用?
a :: String -> IO String
a s = fmap (\x -> x ++ "!") (b s)
b :: String -> IO String
b s = fmap (\x -> '!':x) (fetch s)
fetch :: String -> IO String
fetch s = do
x <- getLine
return $ s ++ x
main = a "hello" >>= putStrLn
我可以看到,如果函数本身不相互依赖,那么重构会简单得多。对于一个简单的例子来说这很好:
a :: String -> String
a s = s ++ "!"
b :: String -> String
b s = '!':s
fetch :: String -> IO String
fetch s = do
x <- getLine
return $ s ++ x
doit :: String -> IO String
doit s = fmap (a . b) (fetch s)
main = doit "hello" >>= putStrLn
但我不知道这在更复杂的程序中是否一定实用。到目前为止,我发现真正隔离像这样的 IO 添加的唯一方法是使用 unsafePerformIO,但是,就其名称而言,如果我能帮助的话,我不想这样做。还有其他方法可以隔离此更改吗?如果重构是实质性的,我会开始倾向于避免这样做(特别是在截止日期等情况下)。
感谢您的建议!
最佳答案
以下是我使用的一些方法。
通过反转控制来减少对效果的依赖。(您在问题中描述的方法之一。)也就是说,在外部执行效果并传递结果(或具有这些结果的函数)部分应用)到纯代码中。不要使用 main
→ a
→ b
→ fetch
,而是使用 main
→ fetch
然后main
→a
→b
:
a :: String -> String
a f = b f ++ "!"
b :: String -> String
b f = '!' : f
fetch :: String -> IO String
fetch s = do
x <- getLine
return $ s ++ x
main = do
f <- fetch "hello"
putStrLn $ a f
对于更复杂的情况,您需要将参数线程化以通过多个级别执行这种“依赖项注入(inject)”,Reader
/ReaderT
可以让您抽象在样板之上。
编写您希望从一开始就需要单子(monad)风格效果的纯代码。(对单子(monad)的选择进行多态。)然后,如果您最终这样做了如果需要在该代码中添加效果,则无需更改实现,只需更改签名。
a :: (Monad m) => String -> m String
a s = (++ "!") <$> b s
b :: (Monad m) => String -> m String
b s = ('!' :) <$> fetch s
fetch :: (Monad m) => String -> m String
fetch s = pure (reverse s)
由于此代码适用于任何具有 Monad
实例(或者实际上只是 Applicative
)的 m
,因此您可以直接在 IO
,或纯粹使用“虚拟”单子(monad)Identity
:
main = putStrLn =<< a "hello"
main = putStrLn $ runIdentity $ a "hello"
然后,当您需要更多效果时,您可以使用“mtl
style”(如 @dfeuer 的答案所述)根据需要启用效果,或者如果您使用相同的 monad到处都是堆栈,只需将 m
替换为该具体类型,例如:
newtype Fetch a = Fetch { unFetch :: IO a }
deriving (Applicative, Functor, Monad, MonadIO)
a :: String -> Fetch String
a s = pure (b s ++ "!")
b :: String -> Fetch String
b s = ('!' :) <$> fetch s
fetch :: String -> Fetch String
fetch s = do
x <- liftIO getLine
return $ s ++ x
main = putStrLn =<< unFetch (a "hello")
mtl
风格的优点是您可以拥有多种不同的效果实现。这使得测试和模拟之类的事情变得容易,因为您可以重用逻辑,但使用不同的“处理程序”运行它以进行生产和测试。事实上,您可以使用代数效果库(例如 freer-effects
)获得更大的灵 active (以牺牲一些运行时性能为代价)。 ,这不仅可以让调用者更改每个效果的处理方式,还可以更改处理效果的顺序。
卷起袖子进行重构。编译器会告诉您任何需要更新的地方。经过足够多次的这样做后,您自然会意识到何时编写需要稍后进行重构的代码,因此您将从一开始就考虑效果,而不会遇到问题。
您对unsafePerformIO
的怀疑是完全正确的!它不仅不安全,因为它破坏了引用透明度,它不安全,因为它还会破坏类型、内存和并发安全性 - 您可以使用它可以将任何类型强制为任何其他类型,导致段错误,或者导致通常不可能发生的死锁和并发错误。您告诉编译器某些代码是纯代码,因此它将假设它可以完成对纯代码所做的所有转换,例如复制、重新排序,甚至删除它,这可能会完全改变您的代码的正确性和性能。代码。
unsafePerformIO 的主要合法用例是使用 FFI 包装外部代码(您知道是纯代码),或者进行特定于 GHC 的性能黑客;否则请远离它,因为它并不意味着普通代码的“逃生舱口”。
关于haskell - 添加 IO 时重构 Haskell,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51201592/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!