- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在探索 Haskell 中的选项,这些选项可以让我将业务逻辑与底层系统的技术实现分开。例如,在 Web 服务器的上下文中,将 Web 服务器处理其接收的信息的方式与其读取和写入数据库的方式分开。要做到这一点,有很多选择,但有两个特别引起了我的注意:释放 Monad 和传递功能记录作为参数。我很难看出其中一种相对于另一种的优缺点。
用于说明我正在讨论的内容的代码片段:
module Lib where
import qualified Control.Monad.Free as FreeMonad
data MyGadt x
= Read (String -> x)
| Write String
x
instance Functor MyGadt where
fmap f (Read g) = Read (f . g)
fmap f (Write str x) = Write str (f x)
programWithFreeMonad :: FreeMonad.Free MyGadt ()
programWithFreeMonad = do
msg <- FreeMonad.liftF $ Read id
FreeMonad.liftF $ Write msg ()
ioInterpreter :: FreeMonad.Free MyGadt x -> IO x
ioInterpreter (FreeMonad.Pure x) = return x
ioInterpreter (FreeMonad.Free (Read f)) = getLine >>= (ioInterpreter . f)
ioInterpreter (FreeMonad.Free (Write str x)) = putStrLn str >> ioInterpreter x
runProgramWithFreeMonad :: IO ()
runProgramWithFreeMonad = ioInterpreter programWithFreeMonad
data Capabilities m = Capabilities
{ myRead :: m String
, myWrite :: String -> m ()
}
programWithCapabilities :: Monad m => Capabilities m -> m ()
programWithCapabilities capabilities = do
msg <- myRead capabilities
myWrite capabilities msg
runProgramWithCapabilities :: IO ()
runProgramWithCapabilities =
programWithCapabilities $ Capabilities {myRead = getLine, myWrite = putStrLn}
这两种解决方案的编写方式不同,因此我认为许多人对其外观以及他们更喜欢哪一种都有自己的看法。但我想知道是否有人对一种解决方案相对于另一种解决方案的优缺点有任何见解。
1 Câu trả lời
即使我们限制自己在自由 monad 和函数记录之间进行选择(不考虑涉及 monad 转换器堆栈和类似 MTL 的类型类的解决方案),仍有很多争论正在进行,而且问题还没有解决。
传统上,简单的 free monad 被指责有两个缺陷:运行时效率低下(这可能重要也可能不重要,这取决于解释器操作的速度相比而言有多慢)和缺乏可扩展性(如何将一个程序提升到另一个程序中)有更丰富的效果吗?)。
"Data types a la carte"首先尝试解决可扩展性问题。后来,"Freer Monads, More Extensible Effects"论文发表,提出了一种更复杂的自由类型来提高单子(monad)绑定(bind)的效率,同时也是定义操作集的可扩展方式。实现这种方法的主要库是:
freer-simple这群人中最容易理解的,显然也是最慢的。好像对括号类型的操作有一些限制。
fused-effects更高效的库,允许括号类型操作。但类型也比较复杂。
polysemy相对较新的库,旨在快速并支持括号类型操作,同时保留简单类型。
这些库的一个吸引人的方面是它们可以让您零碎地解释效果,挑选出一种效果,同时不解释其余的效果。您还可以将抽象效果解释为其他抽象效果,而无需立即转到 IO
.
至于功能记录方法。像 programWithCapability
这样的程序在基本 monad 上是多态的,并且记录由 monad 参数化的函数,在概念上与所谓的 van Laarhoven Free Monad 相关。 :
-- (ops m) is required to be isomorphic to (Π n. i_n -> m j_n)
newtype VLMonad ops a = VLMonad { runVLMonad :: forall m. Monad m => ops m -> m a }
instance Monad (VLMonad ops) where
return a = VLMonad (\_ -> return a)
m >>= f = VLMonad (\ops -> runVLMonad m ops >>= f' ops)
Ở đâu
f' ops a = runVLMonad (f a) ops
来自链接的帖子:
Swierstra notes that by summing together functors representing primitive I/O actions and taking the free monad of that sum, we can produce values use multiple I/O feature sets. Values defined on a subset of features can be lifted into the free monad generated by the sum. The equivalent process can be performed with the van Laarhoven free monad by taking the product of records of the primitive operations. Values defined on a subset of features can be lifted by composing the van Laarhoven free monad with suitable projection functions that pick out the requisite primitive operations.
似乎不存在(?)为您提供预制 VLMonad
类型的库。确实存在的是记录函数但通过 IO 工作的库,例如 RIO
。人们仍然可以对逻辑中的基本 monad 进行抽象,然后在运行逻辑时使用 RIO。或者更喜欢简单性并撕开将 IO
隐藏在逻辑中的多态面纱。
函数记录方法可能具有更容易掌握的优点,是直接在 IO
上工作的增量升级。它也更类似于进行依赖注入(inject)的面向对象方式。
使用唱片本身的人体工程学变得至关重要。目前常用"classy lenses"使程序逻辑独立于具体的记录类型,方便程序的编写。也许有一天也可以使用可扩展记录(就像在更自由的方法中使用可扩展总和类型一样)。
关于haskell - 自由 Monad 与显式传递函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56092971/
Hàm sscanf() trong ngôn ngữ C: đọc tệp tiêu đề dữ liệu theo định dạng được chỉ định từ một chuỗi: ?
Gần đây, tôi gặp phải vấn đề với việc đánh giá trước công việc mà tôi không biết cách giải quyết ngay cả sau khi đã tìm hiểu cách thức hoạt động của từng chức năng. Sau đây là một mã giả. Dưới đây là một hàm có tên là foo() sẽ được truyền một giá trị và trả về một giá trị. Nếu bạn truyền giá trị sau vào hàm foo,
Hàm CStr trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Chuỗi. CStr(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường, bạn có thể
Hàm CSng trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Đơn. CSng(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường,
Hàm CreateObject tạo và trả về tham chiếu đến đối tượng Automation. CreateObject(servername.typename [, location]) Tham số serv
Hàm Cos trả về cosin của một góc. Cos(số) Đối số số có thể là bất kỳ biểu thức số hợp lệ nào thể hiện góc tính bằng radian. Mô tả Hàm Cos lấy một góc và trả về tỷ số giữa hai cạnh của một tam giác vuông. Tỷ lệ này là
Hàm CLng trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con Long. CLng(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường, bạn có thể sử dụng
Hàm CInt trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Số nguyên. CInt(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường,
Hàm Chr trả về ký tự tương ứng với mã ký tự ANSI đã chỉ định. Chr(charcode) Tham số charcode là một số xác định một ký tự. Mô tả Các số từ 0 đến 31 biểu thị tiêu chuẩn không in được
Hàm CDbl trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Double. CDbl(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường, bạn có thể
Hàm CDate trả về một biểu thức đã được chuyển đổi thành Biến thể của kiểu con Date. CDate(date) Tham số ngày là bất kỳ biểu thức ngày hợp lệ nào. Mô tả Hàm IsDate được sử dụng để xác định d
Hàm CCur trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Currency. CCur(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường,
Hàm CByte trả về một biểu thức đã được chuyển đổi thành Biến thể có kiểu con là Byte. CByte(biểu thức) Tham số biểu thức là bất kỳ biểu thức hợp lệ nào. Mô tả Thông thường, bạn có thể
Hàm CBool trả về một biểu thức được chuyển đổi thành một Biến thể có kiểu con Boolean. CBool(expression) expression là bất kỳ biểu thức hợp lệ nào. Nếu cũ
Hàm Atn trả về cung tan của một số. Atn(số) Tham số số có thể là bất kỳ biểu thức số hợp lệ nào. Mô tả Hàm Atn tính tỉ số hai cạnh của một tam giác vuông (số) và trả về cung của góc tương ứng.
Hàm Asc trả về mã ký tự ANSI tương ứng với chữ cái đầu tiên của chuỗi. Asc(chuỗi) Tham số chuỗi là bất kỳ biểu thức chuỗi hợp lệ nào. Nếu đối số chuỗi không chứa ký tự, lỗi thời gian chạy sẽ xảy ra.
Hàm Array trả về một Biến thể chứa một mảng. Mảng(arglist) Đối số arglist là danh sách các giá trị được phân tách bằng dấu phẩy để gán cho các phần tử của mảng có trong Biến thể. Nếu tham số này không được chỉ định,
Hàm Abs trả về giá trị tuyệt đối của một số. Abs(số) Tham số số có thể là bất kỳ biểu thức số hợp lệ nào. Nếu number chứa Null, nó sẽ trả về Null; nếu đó là biến chưa được khởi tạo, nó sẽ trả về 0.
Hàm FormatPercent trả về một biểu thức được định dạng dưới dạng phần trăm (nhân với 100) với dấu % theo sau. Định dạngPhần trăm(biểu thức[,NumDigitsAfterD
Hàm FormatNumber trả về một biểu thức được định dạng dưới dạng số. Định dạngSố(biểu thức [,NumDigitsAfterDecimal [,Inc
Tôi là một lập trình viên xuất sắc, rất giỏi!