如何在签名上匹配空列表类型包装器类型针对类型签名,函数返回 Nil

如何解决如何在签名上匹配空列表类型包装器类型针对类型签名,函数返回 Nil

仍然是关于“PureScript by example”的第 3 章(非相关 previous question)。函数 removeDuplicates 在测试时返回 Nil 并且在 repl 上返回 Nil 而不是 AddressBook 这有点令人失望,因为我期望编译器可以防止这种情况。另一方面,我似乎也无法匹配空地址簿 (type AddressBook = List Entry)。

代码(简化省略不相关部分):

emptyBook :: AddressBook
emptyBook = empty

findEntry :: String -> String -> AddressBook -> Maybe Entry
findEntry firstName lastName = head <<< filter filterEntry
  where
  filterEntry :: Entry -> Boolean
  filterEntry entry = entry.firstName == firstName && entry.lastName == lastName

-- implementation which fails test/breaks compiler warranty

import Data.AddressBook
import Prelude

import Control.Plus (empty)
import Data.List (List(..),filter,foldl,head,null,tail,(:))


removeDuplicates :: AddressBook -> AddressBook
removeDuplicates book = skipIfDup emptyBook book

skipIfDup :: AddressBook -> AddressBook -> AddressBook
skipIfDup newBook Nil = newBook
skipIfDup newBook (entry : book) =
    skipIfDup newerBook book
    where
    newerBook :: AddressBook
    newerBook = 
        case (findEntry entry.firstName entry.lastName newBook) of
            nothing -> newBook
            Just e -> insertEntry e newBook

除了 Nil 之外,我还尝试了 Data.List.nullemptyBook(但我相信符号会反弹),结果相同。使用 [] 不会编译,并将案例带入主案例的 if-else 语句,但随后编译器抱怨未转换 Nil 用例。

这是怎么回事?为什么编译器允许返回 Nil 的函数?我应该如何使用 null 或任何必要的符号正确测试空列表/地址簿的用例?

REPL 输出

> removeDuplicates book with duplicate
Nil

测试用例输出

Failed: Exercise - removeDuplicates because expected ({ address: { city: "Faketown",state: "CA",street: "123 Fake St." },firstName: "John",lastName: "Smith" } : { address: { city: "Arlen",state: "TX",street: "84 Rainey St." },firstName: "Peggy",lastName: "Hill" } : { address: { city: "Springfield",state: "USA",street: "740 Evergreen Terrace" },firstName: "Ned",lastName: "Flanders" } : Nil),got Nil

解决方法

与 C# 或 JavaScript 不同,或者无论您来自何处,PureScript 中的 Nilnull 都不是特殊/神奇的“未初始化引用”一类的东西。 PureScript 根本没有这些。就 PureScript 编译器所知,一切总是“定义”的。

Nil 只是 List 构造函数的名称。看看the List definition

data List a = Nil | Cons a (List a)

它只是一个带有两个构造函数的 sum 类型,其中一个名为 Nil。可以命名为 Abracadabra 并仍然工作相同,但显然库作者希望名称具有某种意义,我猜?愚蠢的人。所以在这种情况下 Nil 应该表示“空列表”。

同样,null 只是一个函数名。 Here's its definition

null :: forall a. List a -> Boolean
null Nil = true
null _ = false

它只返回 true 是给定的列表是空的,否则返回 false


现在,您的情况的实际问题在于:

    newerBook = 
        case (findEntry entry.firstName entry.lastName newBook) of
            Nothing -> newBook
            Just e -> insertEntry e newBook

看看您在做什么:您尝试在 newBook 中查找条目,如果找到,则立即将相同的条目插入回 newBook。这样 newBook 现在会有两个这样的条目。这真的是你的意思吗?

根据我对原始问题的模糊理解,您真正想要做的是将 entry 添加到 newBook,如果它没有在那里。如果它在那里,请不要理会newBook

    newerBook = 
        case (findEntry entry.firstName entry.lastName newBook) of
            Nothing -> entry : newBook
            Just _ -> newBook

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?