- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
所以我一直在阅读一些关于 Haskell(以及其他函数式语言,我想)中的 Zipper 模式来遍历和修改数据结构的内容,我认为这对我来说是一个磨练创建类型技能的好机会Haskell 中的类(class),因为
该类可以为我提供一个通用的遍历接口(interface)来编写代码,而与遍历的数据结构无关。
我想我可能需要两个类——一个用于根数据结构,一个用于创建的特殊数据结构
遍历第一个:
module Zipper where
class Zipper z where
go'up :: z -> Maybe z
go'down :: z -> Maybe z
go'left :: z -> Maybe z
go'right :: z -> Maybe z
class Zippable t where
zipper :: (Zipper z) => t -> z
get :: (Zipper z) => z -> t
put :: (Zipper z) => z -> t -> z
-- store a path through a list, with preceding elements stored in reverse
data ListZipper a = ListZipper { preceding :: [a], following :: [a] }
instance Zipper (ListZipper a) where
go'up ListZipper { preceding = [] } = Nothing
go'up ListZipper { preceding = a:ps, following = fs } =
Just $ ListZipper { preceding = ps, following = a:fs }
go'down ListZipper { following = [] } = Nothing
go'down ListZipper { preceding = ps, following = a:fs } =
Just $ ListZipper { preceding = a:ps, following = fs }
go'left _ = Nothing
go'right _ = Nothing
instance Zippable ([a]) where
zipper as = ListZipper { preceding = [], following = as }
get = following
put z as = z { following = as }
-- binary tree that only stores values at the leaves
data Tree a = Node { left'child :: Tree a, right'child :: Tree a } | Leaf a
-- store a path down a Tree, with branches not taken stored in reverse
data TreeZipper a = TreeZipper { branches :: [Either (Tree a) (Tree a)], subtree :: Tree a }
instance Zipper (TreeZipper a) where
go'up TreeZipper { branches = [] } = Nothing
go'up TreeZipper { branches = (Left l):bs, subtree = r } =
Just $ TreeZipper { branches = bs, subtree = Node { left'child = l, right'child = r } }
go'up TreeZipper { branches = (Right r):bs, subtree = l } =
Just $ TreeZipper { branches = bs, subtree = Node { left'child = l, right'child = r } }
go'down TreeZipper { subtree = Leaf a } = Nothing
go'down TreeZipper { branches = bs, subtree = Node { left'child = l, right'child = r } } =
Just $ TreeZipper { branches = (Right r):bs, subtree = l }
go'left TreeZipper { branches = [] } = Nothing
go'left TreeZipper { branches = (Right r):bs } = Nothing
go'left TreeZipper { branches = (Left l):bs, subtree = r } =
Just $ TreeZipper { branches = (Right r):bs, subtree = l }
go'right TreeZipper { branches = [] } = Nothing
go'right TreeZipper { branches = (Left l):bs } = Nothing
go'right TreeZipper { branches = (Right r):bs, subtree = l } =
Just $ TreeZipper { branches = (Left l):bs, subtree = r }
instance Zippable (Tree a) where
zipper t = TreeZipper { branches = [], subtree = t }
get TreeZipper { subtree = s } = s
put z s = z { subtree = s }
Zippable
都会出现很多这样的错误实例定义:
(Zipper z) =>
声明只想
z
任意
Zipper
.
最佳答案
您还可以使用类型同义词族代替多参数类型类和功能依赖项。在这样的情况下,它们提供了一个更清晰、更易于理解的解决方案。在这种情况下,类和实例将变为:
class Zippable t where
type ZipperType t :: *
enter :: t -> ZipperType t
focus :: ZipperType t -> t
instance Zippable [a] where
type ZipperType [a] = ListZipper a
enter = ...
focus = ...
关于Haskell:为 zipper 创建类型类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/878774/
我一直试图理解为什么我的代码在一个 zip 上不起作用,而在另一个 zip 上不起作用...... THIS zip 解压 , 和 THIS zipper 没有这是我使用的代码: String zip
我最近才开始使用 Java 处理 zip 文件。到目前为止,一切似乎都计划得很好,但我遇到了最后一个障碍:嵌套 zipper 。 我正在尝试搜索具有特定扩展名的文件,以便将它们作为文本文件读取。到目前
我不确定我的命名法在这里是否正确,但我想知道 Haskell 中是否有一个贪婪的 zip 函数。这意味着如果我有 a = [1, 2, 3] b = [4, 5] zip' a b #=> [(Jus
在 Data.Tree.Zipper 中,玫瑰树的 zipper 数据类型是 data TreePos t a = Loc { _content :: t a -- ^ The
我正在比较 Huet's original paper与 Clojure's implementation并试图弄清楚为什么要进行更改。我是 Clojure 新手,所以如果我对 Clojure 代码的
编辑 #2: 这整个问题和探索都是基于我忽略了 zipper 的基本概念;从特定节点的角度来看,它们代表了数据结构中的一个视角。所以 zipper - 在任何时候 - 一对当前节点以及从该节点的角度来
我正在尝试为长度索引列表实现一种 zipper ,它将返回列表的每个项目与删除该元素的列表配对。例如。对于普通列表: zipper :: [a] -> [(a, [a])] zipper = go [
我有两列,第一列的每个单元格包含以逗号分隔的名字列表,第二列的每个单元格包含以逗号分隔的第二名字列表。我需要将两列“压缩”到第三列,其中包含以逗号分隔的全名列表。 例如: Column A
经过几天的努力,我想我终于明白了如何从连续数据创建它们。 但是,搜索了几天后,我似乎找不到任何有关如何将 zipper 转换成其他东西的资源。基本上,我想将一些数据转换为可以传递给 Hiccup 来生
我正在努力解决lens和 zippers 。考虑在 ghci 中运行的以下代码 > import Control.Lens > import Control.Zipper > > :t within
我对分离的 Highcharts 系列之间的间隙有疑问。 例如,我有一个包含两个系列的图表。一个从(2004年3月1日)开始,另一个从(2009年3月1日)开始。它们彼此不重叠。在图表上显示时,条形图
受到最近关于 Haskell 中二维网格的问题的启发,我想知道是否可以创建一个二维 zipper 来跟踪列表列表中的位置。列表上的一维 zipper 使我们能够真正有效地在大型列表中进行本地移动(常见
我在编写可以遍历异构节点树的 zipper 时遇到问题。我有 i) map 列表。每个映射都有 ii) 一个 :inputs 键,其值是映射列表。 我想使用 zipper 来访问每一片叶子并添加一个计
repl> (-> root zip/down zip/right) [{:answer-keys [5 6], :id 3} {:l [{:id 2, :answer-keys []}], :pno
我有大量(比如 1000 个)可变大小(比如从 4k 到 400k)的 zip,一个 zip 条目。我设法以经典方式解压所有内容,但性能并不真正令人满意...您是否知道可以帮助提高性能的其他方式(ni
我有以下无法编译的代码。 fn main() { let a = "123" .chars() .chain("4566".chars()) .zip(
在为期末考试做一些动态规划问题的练习时,我发现这个问题难倒了我。 Zippers:给定三个字符串,你要判断是否可以将前两个字符串中的字符组合成第三个字符串。前两个字符串可以任意混合,但每个字符在第三个
我有两个要压缩的列表: a = ((1,2,3), (4,5,6), ... ) h = ('a','b', ... ) 一个简单的 zip(a,h) 返回: [ ((1,2,3),'a'), ((4
我正在上传一个 zip 文件,然后我需要在具有相同文件名的路径中提取该文件。所以我使用了Chumper/Zipper为了完成我的任务,我使用了以下代码,即 $zipper->zip('storage
我正在学习镜头包。我必须说这是一项相当具有挑战性的任务。 有人能告诉我如何用镜头的 zipper 穿过一棵树吗?特别是,我如何编写一个函数来获取根列表并允许我访问子树的分支? 假设我有这棵树。如果我的
我是一名优秀的程序员,十分优秀!