- 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/
SQLite、Content provider 和 Shared Preference 之间的所有已知区别。 但我想知道什么时候需要根据情况使用 SQLite 或 Content Provider 或
警告:我正在使用一个我无法完全控制的后端,所以我正在努力解决 Backbone 中的一些注意事项,这些注意事项可能在其他地方更好地解决......不幸的是,我别无选择,只能在这里处理它们! 所以,我的
我一整天都在挣扎。我的预输入搜索表达式与远程 json 数据完美配合。但是当我尝试使用相同的 json 数据作为预取数据时,建议为空。点击第一个标志后,我收到预定义消息“无法找到任何内容...”,结果
我正在制作一个模拟 NHL 选秀彩票的程序,其中屏幕右侧应该有一个 JTextField,并且在左侧绘制弹跳的选秀球。我创建了一个名为 Ball 的类,它实现了 Runnable,并在我的主 Draf
这个问题已经有答案了: How can I calculate a time span in Java and format the output? (18 个回答) 已关闭 9 年前。 这是我的代码
我有一个 ASP.NET Web API 应用程序在我的本地 IIS 实例上运行。 Web 应用程序配置有 CORS。我调用的 Web API 方法类似于: [POST("/API/{foo}/{ba
我将用户输入的时间和日期作为: DatePicker dp = (DatePicker) findViewById(R.id.datePicker); TimePicker tp = (TimePic
放宽“邻居”的标准是否足够,或者是否有其他标准行动可以采取? 最佳答案 如果所有相邻解决方案都是 Tabu,则听起来您的 Tabu 列表的大小太长或您的释放策略太严格。一个好的 Tabu 列表长度是
我正在阅读来自 cppreference 的代码示例: #include #include #include #include template void print_queue(T& q)
我快疯了,我试图理解工具提示的行为,但没有成功。 1. 第一个问题是当我尝试通过插件(按钮 1)在点击事件中使用它时 -> 如果您转到 Fiddle,您会在“内容”内看到该函数' 每次点击都会调用该属
我在功能组件中有以下代码: const [ folder, setFolder ] = useState([]); const folderData = useContext(FolderContex
我在使用预签名网址和 AFNetworking 3.0 从 S3 获取图像时遇到问题。我可以使用 NSMutableURLRequest 和 NSURLSession 获取图像,但是当我使用 AFHT
我正在使用 Oracle ojdbc 12 和 Java 8 处理 Oracle UCP 管理器的问题。当 UCP 池启动失败时,我希望关闭它创建的连接。 当池初始化期间遇到 ORA-02391:超过
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve
引用这个plunker: https://plnkr.co/edit/GWsbdDWVvBYNMqyxzlLY?p=preview 我在 styles.css 文件和 src/app.ts 文件中指定
为什么我的条形这么细?我尝试将宽度设置为 1,它们变得非常厚。我不知道还能尝试什么。默认厚度为 0.8,这是应该的样子吗? import matplotlib.pyplot as plt import
当我编写时,查询按预期执行: SELECT id, day2.count - day1.count AS diff FROM day1 NATURAL JOIN day2; 但我真正想要的是右连接。当
我有以下时间数据: 0 08/01/16 13:07:46,335437 1 18/02/16 08:40:40,565575 2 14/01/16 22:2
一些背景知识 -我的 NodeJS 服务器在端口 3001 上运行,我的 React 应用程序在端口 3000 上运行。我在 React 应用程序 package.json 中设置了一个代理来代理对端
我面临着一个愚蠢的问题。我试图在我的 Angular 应用程序中延迟加载我的图像,我已经尝试过这个2: 但是他们都设置了 src attr 而不是 data-src,我在这里遗漏了什么吗?保留 d
我是一名优秀的程序员,十分优秀!