- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我对 Haskell 中的类感到困惑,如下所示。
我可以定义一个采用 Integral 参数的函数,并成功为其提供 Num 参数:
gi :: Integral a => a -> a
gi i = i
gin = gi (3 :: Num a => a)
我可以定义一个采用 Num 参数的函数,并成功为其提供 Integral 参数:
fn :: Num a => a -> a
fn n = n
fni = fn (3 :: Integral a => a)
我可以定义一个 Integral 值并为其分配一个 Num
i :: Integral a => a
i = (3 :: Num a => a)
但是,如果我尝试定义一个 Num 值,然后为其分配一个 Integral 值,则会出现解析错误
- this doesn't work
n :: Num a => a
n = (3 :: Integral a => a)
也许我对我的面向对象背景感到困惑。但是为什么函数变量似乎让你“双向”,即当“期望”父类(super class)时可以提供子类的值,并且当需要子类时可以提供父类(super class)的值,而在赋值时你可以提供将父类(super class)分配给子类值,但无法将子类分配给父类(super class)值?
作为比较,在面向对象编程中,您通常可以将子值分配给父类型,但反之则不然。在 Haskell 中,第二对示例的情况似乎相反。
最佳答案
前两个示例实际上与 Num
和 Integral
之间的关系没有任何关系。
看一下 gin
和 fni
的类型。让我们一起来做吧:
> :t gin
gin :: Integer
> :t fni
fni :: Integer
发生什么事了?这称为“类型默认”。
从技术上讲,Haskell 中的任何数字文字,如 3
或 5
或 42
都具有类型 Num a => a
.因此,如果您希望它只是一个整数,该死的,您必须始终编写 42::Integer
而不是 42
。这非常不方便。
因此,为了解决这个问题,Haskell 有某些规则,在某些特殊情况下,规定当类型变为通用类型时要替换的具体类型。如果同时使用 Num
和 Integral
,默认类型为 Integer
。
因此,当编译器看到 3
并将其用作 gi
的参数时,编译器默认为 Integer
。就是这样。您对 Num a
的附加约束不会产生进一步的影响,因为 Integer
实际上已经是 Num
的实例。
另一方面,对于最后两个示例,区别在于您显式指定了类型签名。你不只是把它留给编译器来决定,不!您特别提到了 n::Num a => a
。因此编译器无法再决定 n::Integer
。它必须是通用的。
由于它是通用的,并且被限制为 Num
,因此 Integral
类型不起作用,因为,正如您所正确指出的,Num
code> 不是 Integral 的子类。
您可以通过为 fni
提供类型签名来验证这一点:
-- no longer works
fni :: Num a => a
fni = fn (3 :: Integral a => a)
等等,但是 n
不应该仍然有效吗?毕竟,在面向对象中这会很好地工作。以 C# 为例:
class Num {}
class Integral : Num {}
class Integer : Integral {}
Num a = (Integer)3
// ^ this is valid (modulo pseudocode), because `Integer` is a subclass of `Num`
啊,但这不是泛型类型!在上面的示例中,a
是具体类型 Num
的值,而在 Haskell 代码中 a
本身就是一种类型,但仅限于是Num
。这更像是 C# 接口(interface),而不是 C# 类。
泛型类型(无论是否在 Haskell 中)实际上是相反的!取这样的值:
x :: a
x = ...
这个类型签名的意思是“谁需要x
,就来拿吧!但是首先命名一个类型a
。然后是值 x
将属于该类型。无论您命名哪种类型,x
都将是“
或者,更简单地说,选择泛型类型的是函数的调用者(或值的使用者),而不是实现者。
因此,如果您说 n::Num a => a
,则意味着值 n
必须能够“变形”为任何 类型 a
无论如何,只要该类型有一个 Num
实例。无论谁在计算中使用n
- 这个人都会选择a
是什么。您,n
的实现者,无权选择它。
由于您无法选择 a
是什么,因此您无法将其缩小为不仅仅是任何 Num
,而是一个 >整体
。因为,你知道,有一些 Num
不是 Integral
,所以如果使用 n
的人选择,你会怎么做那些非Integral
类型之一是a
?
对于i
,这工作得很好,因为每个Integral
也必须是Num
,所以无论i的使用者是什么
选择 a
,您肯定知道它将是 Num
。
关于haskell - 与 Haskell 类的混淆,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67732331/
在 Haskell 中,类型声明使用双冒号,即 (::),如 not::Bool -> Bool。 但是在许多语法与 Haskell 类似的语言中,例如榆树、 Agda 、他们使用单个冒号(:)来声明
insertST :: StateDecoder -> SomeState -> Update SomeState SomeThing insertST stDecoder st = ... Stat
如果这个问题有点含糊,请提前道歉。这是一些周末白日梦的结果。 借助 Haskell 出色的类型系统,将数学(尤其是代数)结构表达为类型类是非常令人愉快的。我的意思是,看看 numeric-prelud
我有需要每 5 分钟执行一次的小程序。 目前,我有执行该任务的 shell 脚本,但我想通过 CLI 中的键为用户提供无需其他脚本即可运行它的能力。 实现这一目标的最佳方法是什么? 最佳答案 我想你会
RWH 面世已经有一段时间了(将近 3 年)。在在线跟踪这本书的渐进式写作之后,我渴望获得我的副本(我认为这是写书的最佳方式之一。)在所有相当学术性的论文中,作为一个 haskell 学生,读起来多么
一个经典的编程练习是用 Lisp/Scheme 编写一个 Lisp/Scheme 解释器。可以利用完整语言的力量来为该语言的子集生成解释器。 Haskell 有类似的练习吗?我想使用 Haskell
以下摘自' Learn You a Haskell ' 表示 f 在函数中用作“值的类型”。 这是什么意思?即“值的类型”是什么意思? Int 是“值的类型”,对吗?但是 Maybe 不是“值的类型”
现在我正在尝试创建一个基本函数,用于删除句子中的所有空格或逗号。 stringToIntList :: [Char] -> [Char] stringToIntList inpt = [ a | a
我是 Haskell 的新手,对模式匹配有疑问。这是代码的高度简化版本: data Value = MyBool Bool | MyInt Integer codeDuplicate1 :: Valu
如何解释这个表达式? :t (+) (+3) (*100) 自 和 具有相同的优先级并且是左结合的。我认为这与 ((+) (+3)) (*100) 相同.但是,我不知道它的作用。在 Learn
这怎么行 > (* 30) 4 120 但这不是 > * 30 40 error: parse error on input ‘*’ 最佳答案 (* 30) 是一个 section,它仍然将 * 视为
我想创建一个函数,删除满足第二个参数中给定谓词的第一个元素。像这样: removeFirst "abab" ( 'b') = "abab" removeFirst [1,2,3,4] even =
Context : def fib(n): if n aand returns a memoized version of the same function. The trick is t
我明白惰性求值是什么,它是如何工作的以及它有什么优势,但是你能解释一下 Haskell 中什么是严格求值吗?我似乎找不到太多关于它的信息,因为惰性评估是最著名的。 他们各自的优势是什么。什么时候真正使
digits :: Int -> [Int] digits n = reverse (x) where x | n digits 1234 = [3,1,2,4]
我在 F# 中有以下代码(来自一本书) open System.Collections.Generic type Table = abstract Item : 'T -> 'U with ge
我对 Haskell 比较陌生,过去几周一直在尝试学习它,但一直停留在过滤器和谓词上,我希望能得到帮助以帮助理解。 我遇到了一个问题,我有一个元组列表。每个元组包含一个 (songName, song
我是 haskell 的初学者,我试图为埃拉托色尼筛法定义一个简单的函数,但它说错误: • Couldn't match expected type ‘Bool -> Bool’
我是 Haskell 语言的新手,我在使用 read 函数时遇到了一些问题。准确地说,我的理解是: read "8.2" + 3.8 应该返回 12.0,因为我们希望返回与第二个成员相同的类型。我真正
当我尝试使用真实项目来驱动它来学习 Haskell 时,我遇到了以下定义。我不明白每个参数前面的感叹号是什么意思,我的书上好像也没有提到。 data MidiMessage = MidiMessage
我是一名优秀的程序员,十分优秀!