- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
考虑下一个例子。我有一个 monad MyM
,它只是一个 StateT
{-# LANGUAGE TypeFamilies #-}
import Control.Monad.State
import Control.Monad.Reader
type MyS = Int
type MyM = StateT MyS
通常MyM
用于读写MyS
状态,所以我有如下函数:
f1 :: (MonadState m, StateType m ~ MyS) => m ()
f1 = modify (+1)
但有时我只需要读取 MyS
,所以我想要 MonadReader
上下文而不是 MonadState
:
f2 :: (MonadReader m, EnvType m ~ MyS) => m Int
f2 = liftM (+1) ask
我想写这样的东西:
f3 :: (MonadState m, StateType m ~ MyS) => m Int
f3 = f1 >> f2
所以基本上我需要每个 MonadState
实例也是具有相应系列类型的 MonadReader
实例。有点像
instance MonadState m => MonadReader where
type EnvType m = StateType m
...
但我找不到如何进行类型检查的方法。是否可以表达MonadState
和MonadReader
之间的这种关系?
谢谢。
最佳答案
听起来您想要的本质上是 ask
与 get
具有相同的效果.我不禁想知道你为什么不直接使用 get
在那种情况下:)
如果您的目标是根据可用的内容编写读取状态或环境的代码,您必须询问您打算做什么,比如 ReaderT r (StateT s m) a
,你有两个。因此,您不能只做:
instance MonadState m => MonadReader m where
type EnvType m = StateType m
ask = get
因为您会与现有实例发生冲突。但是,您可以:
{-# LANGUAGE GeneralizedNewtypeDeriving, TypeFamilies #-}
newtype ReadState m a = RS { unRS :: m a }
deriving (Monad)
instance MonadState m => MonadReader (ReadState m) where
type EnvType (ReadState m) = StateType m
ask = RS get
local f (RS m) = RS $ do
s <- get
modify f
x <- m
put s
return x
然后如果你有一个像f2
这样的多态读取器值, 你可以拉一个 MonadState
用unRS
出来.如果你想使用一些更狡猾的扩展,试试 RankNTypes
:
useStateAsEnv :: (MonadState n) => (forall m . (MonadReader m, EnvType m ~ StateType n) => m a) -> n a
useStateAsEnv m = unRS m
然后你可以做 useStateAsEnv (liftM (+1) ask)
得到 MonadState
值(value)。然而,我还没有彻底研究这在实践中有多大用处——生成 forall m. MonadReader m => m a
类型的值。 ,你几乎只能使用 ask
, local
, 和 monadic 函数。
这是一个类似的,不太通用但可能更有用的东西,使用标准转换器:
readerToState :: (Monad m) => ReaderT r m a -> StateT r m a
readerToState reader = StateT $ \env -> do
res <- runReaderT reader env
return (res, env)
编辑:稍后再考虑这个问题,您可能可以将其推广到具有 lift . runReaderT reader =<< get
的任何状态 monad 转换器。 , 但类型开始变得相当笨拙:
:: (Monad m, MonadTrans t, MonadState (t m)) => ReaderT (StateType (t m)) m b -> t m b
是以上内容的概括,但实际上可能没有用。
关于haskell - 单子(monad)-tf : MonadReader instance for MonadState,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7292766/
我知道我的问题有点含糊,但我不知道如何描述它。我问过很多地方,但似乎没有人理解我为什么要这样做。但请耐心等待,我会解释为什么我想要这样的东西。 我使用 Liquid Templates 允许用户在我的
这个问题在这里已经有了答案: what is the difference between null != object and object!=null [duplicate] (2 个回答) 7年
当我在我的本地主机 Google App Engine 应用程序中将日志记录级别更改为 FINE 时,我开始在我的跟踪堆栈中看到这些: Apr 17, 2013 4:54:20 PM com.goog
Python 有内置函数 type : class type(object) With one argument, return the type of an object. The return v
我正在使用深度学习进行语义分割,我遇到了以下术语:语义分割、实例检测、对象检测 和对象分割. 它们有什么区别? 最佳答案 这些术语的某些用法对用户而言是主观的或依赖于上下文,但据我所知对这些术语的合理
我面临 -[NSConcreteMutableData release] 的问题:消息发送到已释放的实例,我也附上了我的示例代码。 - (IBAction)uploadImage { NSString
我试图显示模型中的单个实例(数据库行),其中多个实例共享多行的相同字段(列)值。为了澄清这一说法,我有以下情况: ID/Title/Slug/Modified 1 Car A 1s ag
我正在尝试使用mockito来模拟服务。然而,我没有找到一种方法来告诉mockito,给定一个类的实例返回给我相同的实例: 类似于: given(service.add(any(Individua
我知道如何从父类(super class)原型(prototype)创建子类原型(prototype)。但是,如果我已经有了父类(super class)对象的实例来创建子类对象怎么办? 在 JS 中
鉴于 Kotlin 1.1。对于某个类的 instance,instance::class.java 和 instance.javaClass 似乎几乎是等价的: val i = 0 println(
这个问题在这里已经有了答案: 8年前关闭。 Possible Duplicate: Find out the instance id from within an ec2 machine 我正在寻找从
为什么我的 Instantiate 函数没有创建 That 的“空白”实例? 我有以下最小类: classdef That < handle properties This = '' end
Session session = HibernateUtil.getSessionFactory().openSession(); Transaction tx = session.beginTra
考虑以下几点: public class A { public String name = "i am a A instance"; } public class B extends A {
我正在使用 Scalr 来扩展网站服务器。 在 Apache 服务器上,我安装了 Sakai,并为 Linux 机器创建了一个启动脚本。 问题是,如何确保MySQL实例在Apache服务器启动之前启动
Android Realm DB 允许使用 Realm.getInstance() 获取多个实例。这些中的最佳实践是什么? :1.创建单个实例(应用程序范围)并在任何地方使用它2. 在需要时获取一个新
我很难理解为什么修改实例 a 中的属性会修改实例 b 中的相同属性。 var A = function (){ }; A.prototype.data = { value : 0 }; var
我将 Weka 用作更长管道的一部分,因此,我无法承受将所有数据写入文件或数据库只是为了创建一个 Instances。目的。我可以即时做的是创建 Instance 的列表对象。 来自 this pag
class C: def func(self, a): print(a) c = C() print(c.__dict__) # {} c.func = c.func # c.func i
Angular Routing 文档提到了组件实例创建、组件实例激活和路由激活。 文档没有解释这些概念的区别,以及每次创建/激活发生的时间。 问题 实例创建和实例激活有什么区别? 实例激活和路由激活有
我是一名优秀的程序员,十分优秀!