- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
当进行元编程时,将有关程序已知但在Hindley-Milner中不可推断的类型的信息传递给Haskell的类型系统信息可能是有用的(或必要的)。在Haskell中是否有一个库(或语言扩展等)提供执行此操作的工具(即程序类型注释)?
考虑一种情况,您正在使用异构列表(例如,使用Data.Dynamic
库或存在性量化实现),并且希望将列表过滤为沼泽标准,同类型的Haskell列表。你可以写一个像
import Data.Dynamic
import Data.Typeable
dynListToList :: (Typeable a) => [Dynamic] -> [a]
dynListToList = (map fromJust) . (filter isJust) . (map fromDynamic)
foo :: [Int]
foo = dynListToList [ toDyn (1 :: Int)
, toDyn (2 :: Int)
, toDyn ("foo" :: String) ]
foo
是列表
[1, 2] :: [Int]
;效果很好,您又回到了Haskell的字体系统可以完成其任务的坚实基础。
dynListToList
所产生的列表的类型是什么,但是(b)您的程序确实包含解决此问题所需的信息,只有(c)它不是类型系统可访问的形式。
Data.Typeable
提供的类型检查工具,您的程序具有执行此操作所需的所有信息,但是据我所知-这是问题的本质-无法将其传递给类型系统。这是一些伪Haskell,它说明了我的意思:
import Data.Dynamic
import Data.Typeable
randList :: (Typeable a) => [Dynamic] -> IO [a]
randList dl = do
tr <- randItem $ map dynTypeRep dl
return (dynListToList dl :: [<tr>]) -- This thing should have the type
-- represented by `tr`
randItem
从列表中选择一个随机项。)
return
的参数上没有类型注释的情况下,编译器将告诉您它具有“模糊类型”,并要求您提供一个。但是您不能提供手动类型注释,因为在写时类型未知(并且可能会有所不同)。但是,该类型在运行时是已知的-尽管是类型系统无法使用的形式(此处,所需的类型由值
tr
,
TypeRep
表示-有关详细信息,请参见
Data.Typeable
)。
:: [<tr>]
是我想要发生的魔术。有没有办法以编程方式为类型系统提供类型信息;也就是说,程序的值中包含类型信息?
??? -> TypeRep -> a
的函数,该函数采用Haskell的类型系统未知的类型的值和
TypeRep
并说:“相信我,编译器,我知道我在做什么。此
TypeRep
表示的值。” (请注意,这不是
unsafeCoerce
所做的。)
最佳答案
不,你不能这样做。它的长处和短处是您正在尝试编写一个依赖类型的函数,而Haskell并不是一种依赖类型的语言。您无法将TypeRep
值提升为真实类型,因此无法写下所需函数的类型。为了更详细地解释这一点,我首先要说明为什么用短语randList
类型表述的方式实际上没有意义。然后,我将解释为什么您不能做自己想做的事情。最后,我将简要提及一些有关实际操作的想法。
存在的
您对randList
的类型签名不能表示您想要的含义。请记住,Haskell中的所有类型变量都是通用量化的,
randList :: forall a. Typeable a => [Dynamic] -> IO [a]
randList dyns :: IO [Int]
。我必须能够为所有
a
提供返回值,而不仅仅是为某些
a
提供返回值。将其视为一种游戏,它是调用者可以选择
a
而不是函数本身的游戏。您想要说的是什么(这不是有效的Haskell语法,尽管您可以通过使用存在数据类型1将其转换为有效的Haskell)。
randList :: [Dynamic] -> (exists a. Typeable a => IO [a])
a
,它是
Typeable
的一个实例,但不一定是任何此类类型。但是即使这样,您仍然会有两个问题。首先,即使您可以构建这样的列表,也可以使用它做些什么?其次,事实证明,您甚至根本无法构建它。
Typeable
的实例,因此您可以如何使用它们?
Looking at the documentation,我们看到只有两个函数2接受
Typeable
的实例:
typeOf :: Typeable a => a -> TypeRep
,来自类型类本身(实际上是其中的唯一方法);和cast :: (Typeable a, Typeable b) => a -> Maybe b
(已使用unsafeCoerce
实现,不能用其他方式编写)。 typeOf
和
cast
。由于我们永远无法对它们进行任何其他有用的操作,因此我们的存在可能同样会(同样,也是无效的Haskell)
randList :: [Dynamic] -> IO [(TypeRep, forall b. Typeable b => Maybe b)]
typeOf
和
cast
应用于列表的每个元素,存储结果并丢弃现在无用的现有类型原始值,这就是我们得到的。显然,此列表的
TypeRep
部分没有用。列表的后半部分也不是。由于我们回到了通用量化的类型,因此
randList
的调用方再次有权要求他们为他们选择的任何(可键入的)
Maybe Int
获取
Maybe Bool
,
Maybe b
或
b
。 (实际上,它们具有比以前更大的功能,因为它们可以将列表的不同元素实例化为不同的类型。)但是,除非他们已经知道,否则他们无法弄清楚它们从哪个类型转换而来。丢失了您要保留的类型信息。
return $ dynListToList dl
)时出现错误。您以哪种特定类型调用
dynListToList
?回想一下
dynListToList :: forall a. Typeable a => [Dynamic] -> [a]
;因此,
randList
负责选择要使用的
a
。但是它不知道选择哪一个
dynListToList
。再次,这就是问题的根源!因此,您尝试返回的类型是未指定的,因此模棱两可。3
a
,我们还有其
a
。因此,也许我们可以将其打包:
randList :: [Dynamic] -> (exists a. Typeable a => IO (TypeRep,[a]))
TypeRep
和
TypeRep
根本没有链接。这正是您要表达的内容:链接
[a]
和
TypeRep
的某种方法。
toType :: TypeRep -> *
a
是所有类型的类型。如果您以前从未看过种类,那么它们就是要键入值。
*
对类型进行分类,
*
对单参数类型构造函数进行分类,等等。(例如
* -> *
,
Int :: *
,
Maybe :: * -> *
和
Either :: * -> * -> *
。)
randList :: [Dynamic] -> (exists (tr :: TypeRep).
Typeable (toType tr) => IO (tr, [toType tr]))
randList dl = do
tr <- randItem $ map dynTypeRep dl
return (tr, dynListToList dl :: [toType tr])
-- In fact, in an ideal world, the `:: [toType tr]` signature would be
-- inferable.
Maybe Int :: *
以使其对应的类型对列表元素进行分类。如果您能做到这一点,您就会被设置。但是在Haskell中完全不可能编写
TypeRep
:这样做需要一种依赖类型的语言,因为
toType :: TypeRep -> *
是一种依赖于值的类型。
toType tr
取决于值
head "abc"
。同样,我们有类型构造函数,因此类型依赖于其他类型是可以接受的。考虑
"abc"
及其依赖于
Maybe Int
的方式。我们甚至可以拥有取决于类型的值!考虑
Int
。这实际上是一组函数:
id :: a -> a
,
id_Int :: Int -> Int
等。我们拥有的函数取决于
id_Bool :: Bool -> Bool
的类型。 (实际上是
a
;尽管我们无法在Haskell中编写此代码,但是有些语言可以。)
id = \(a :: *) (x :: a) -> x
,即长度为7的整数列表的类型。在这里,
Vec 7 Int
:一种类型,其第一个参数必须是
Vec :: Nat -> * -> *
类型的值。但是我们不能在Haskell中写这种东西。4支持这种语言的语言称为依赖类型的语言(这将使我们像上面那样编写
Nat
);示例包括
Coq和
Agda。 (这类语言通常兼作证明助手,通常用于研究工作,而不是编写实际的代码。从属类型很难,使它们对日常编程有用是研究的活跃领域。)
id
易于实现(在操作上)并且完全不安全的原因:在运行时,它是空操作,但它属于类型系统。因此,在Haskell类型系统中完全不可能实现
unsafeCoerce
之类的东西。
toType
。对于某些问题,您可以避免这种情况。我们可以写下函数的类型,但只能通过作弊来实现。这就是
unsafeCoerce
的工作原理。但是正如我们在上面看到的,Haskell内部甚至没有一个很好的类型可以解决这个问题。虚构的
fromDynamic
函数允许您为程序指定类型,但您甚至无法写下
toType
的类型!
toType
和
Typeable
并没有显示太多。 (正如他们所说,也许您是“用Python的口音讲Haskell”。)如果您只需要处理一组有限的数据类型,则可以将其 bundle 到一个普通的旧代数数据类型中:
data MyType = MTInt Int | MTBool Bool | MTString String
Dynamic
,而只需使用
isMTInt
或
filter isMTInt
即可。
filter (isSameMTAs randomMT)
编码。但是坦率地说,除非您真的,真的,真的,真的,真的,真的知道自己在做什么,否则这不是一个好主意。即使这样,也可能不是。如果您需要
unsafeCoerce
,就会知道,这不只是一件方便的事情。
{-# LANGUAGE ExistentialQuantification #-}
data TypeableList = forall a. Typeable a => TypeableList [a]
randList :: [Dynamic] -> IO TypeableList
unsafeCoerce
编写起来更清晰。
exists
和
toDyn :: Typeable a => a -> Dynamic
。但是,
fromDyn :: Typeable a => Dynamic -> a -> a
或多或少是
Dynamic
周围的存在性包装器,依靠
Typeable
和
typeOf
s知道何时应使用
TypeRep
(GHC使用某些实现特定的类型和
unsafeCoerce
,但是您可以通过这种方式做到这一点,但
unsafeCoerce
可能是例外/
dynApply
),因此
dynApp
不会执行任何新操作。而且
toDyn
并不真正期望其类型为
fromDyn
的参数;它只是
a
的包装。这些功能以及其他类似功能无法提供
cast
和
typeOf
所无法提供的任何额外功能。 (例如,返回
cast
对您的问题不是很有用!)
{-# LANGUAGE ExistentialQuantification #-}
import Data.Dynamic
import Data.Typeable
import Data.Maybe
randItem :: [a] -> IO a
randItem = return . head -- Good enough for a short and non-compiling example
dynListToList :: Typeable a => [Dynamic] -> [a]
dynListToList = mapMaybe fromDynamic
data TypeableList = forall a. Typeable a => TypeableList [a]
randList :: [Dynamic] -> IO TypeableList
randList dl = do
tr <- randItem $ map dynTypeRep dl
return . TypeableList $ dynListToList dl -- Error! Ambiguous type variable.
SO12273982.hs:17:27:
Ambiguous type variable `a0' in the constraint:
(Typeable a0) arising from a use of `dynListToList'
Probable fix: add a type signature that fixes these type variable(s)
In the second argument of `($)', namely `dynListToList dl'
In a stmt of a 'do' block: return . TypeableList $ dynListToList dl
In the expression:
do { tr <- randItem $ map dynTypeRep dl;
return . TypeableList $ dynListToList dl }
Dynamic
示例仍不存在)5,但是您将能够通过使用看起来像值的非常有表现力的类型来编写
TypeRep -> *
。
Vec
。但是,这采用了升级后的
type family ToType :: TypeRep -> *
类型的类型,而不是
TypeRep
类型的值;而且,您仍然无法实现它。 (至少我不这么认为,而且我不知道你会怎么做,但是我不是这方面的专家。)但是目前,我们还很遥远。
关于haskell - Haskell中的程序化类型注释,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12273982/
我想做的是让 JTextPane 在 JPanel 中占用尽可能多的空间。对于我使用的 UpdateInfoPanel: public class UpdateInfoPanel extends JP
我在 JPanel 中有一个 JTextArea,我想将其与 JScrollPane 一起使用。我正在使用 GridBagLayout。当我运行它时,框架似乎为 JScrollPane 腾出了空间,但
我想在 xcode 中实现以下功能。 我有一个 View Controller 。在这个 UIViewController 中,我有一个 UITabBar。它们下面是一个 UIView。将 UITab
有谁知道Firebird 2.5有没有类似于SQL中“STUFF”函数的功能? 我有一个包含父用户记录的表,另一个表包含与父相关的子用户记录。我希望能够提取用户拥有的“ROLES”的逗号分隔字符串,而
我想使用 JSON 作为 mirth channel 的输入和输出,例如详细信息保存在数据库中或创建 HL7 消息。 简而言之,输入为 JSON 解析它并输出为任何格式。 最佳答案 var objec
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个以前可能被问过的问题,但我很难找到正确的描述。我希望有人能帮助我。 在下面的代码中,我设置了varprice,我想添加javascript变量accu_id以通过rails在我的数据库中查找记
我有一个简单的 SVG 文件,在 Firefox 中可以正常查看 - 它的一些包装文本使用 foreignObject 包含一些 HTML - 文本包装在 div 中:
所以我正在为学校编写一个 Ruby 程序,如果某个值是 1 或 3,则将 bool 值更改为 true,如果是 0 或 2,则更改为 false。由于我有 Java 背景,所以我认为这段代码应该有效:
我做了什么: 我在这些账户之间创建了 VPC 对等连接 互联网网关也连接到每个 VPC 还配置了路由表(以允许来自双方的流量) 情况1: 当这两个 VPC 在同一个账户中时,我成功测试了从另一个 La
我有一个名为 contacts 的表: user_id contact_id 10294 10295 10294 10293 10293 10294 102
我正在使用 Magento 中的新模板。为避免重复代码,我想为每个产品预览使用相同的子模板。 特别是我做了这样一个展示: $products = Mage::getModel('catalog/pro
“for”是否总是检查协议(protocol)中定义的每个函数中第一个参数的类型? 编辑(改写): 当协议(protocol)方法只有一个参数时,根据该单个参数的类型(直接或任意)找到实现。当协议(p
我想从我的 PHP 代码中调用 JavaScript 函数。我通过使用以下方法实现了这一点: echo ' drawChart($id); '; 这工作正常,但我想从我的 PHP 代码中获取数据,我使
这个问题已经有答案了: Event binding on dynamically created elements? (23 个回答) 已关闭 5 年前。 我有一个动态表单,我想在其中附加一些其他 h
我正在尝试找到一种解决方案,以在 componentDidMount 中的映射项上使用 setState。 我正在使用 GraphQL连同 Gatsby返回许多 data 项目,但要求在特定的 pat
我在 ScrollView 中有一个 View 。只要用户按住该 View ,我想每 80 毫秒调用一次方法。这是我已经实现的: final Runnable vibrate = new Runnab
我用 jni 开发了一个 android 应用程序。我在 GetStringUTFChars 的 dvmDecodeIndirectRef 中得到了一个 dvmabort。我只中止了一次。 为什么会这
当我到达我的 Activity 时,我调用 FragmentPagerAdapter 来处理我的不同选项卡。在我的一个选项卡中,我想显示一个 RecyclerView,但他从未出现过,有了断点,我看到
当我按下 Activity 中的按钮时,会弹出一个 DialogFragment。在对话框 fragment 中,有一个看起来像普通 ListView 的 RecyclerView。 我想要的行为是当
我是一名优秀的程序员,十分优秀!