首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >为二进制搜索树定义fmap

为二进制搜索树定义fmap
EN

Stack Overflow用户
提问于 2014-04-06 12:36:59
回答 5查看 1.5K关注 0票数 3

我正在完成“Haskell开始”一书中的练习。练习4-8是使二叉树成为函式的一个实例,并定义fmap.这就是这棵树的样子:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
data BinaryTree a = Node a (BinaryTree a) (BinaryTree a) 
                  | Leaf
                    deriving Show

因为它是一个搜索树,所以树上的所有操作都必须保持不变,即左侧子树中的所有值都是<节点的值,而右侧子树中的所有值都是>节点的值。这意味着树中的所有值都必须是序号(Ord a => BinaryTree a)。

两个问题:

  1. fmap :: (a -> b) -> BinaryTree a -> BinaryTree b以来,我如何执行b也是序数?如果它不一定是函子,我可以简单地做fmapOrd :: (Ord a, Ord b) => (a -> b) -> BinaryTree a -> BinaryTree b,但是函子类型不强制执行Ord约束。
  2. 高效的实现是什么样子的?我的第一个想法是折叠在树上,并根据映射的值建立一棵新的树。不幸的是,我没有走这么远,因为(1)。
EN

回答 5

Stack Overflow用户

回答已采纳

发布于 2014-04-06 12:59:25

Functorfmap的要点是,它适用于可以存储在数据结构中的所有ab,就像Monad也适用于所有类型的a一样。您的Functor实例应该如下所示

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
instance Functor BinaryTree where
    fmap f Leaf = Leaf
    fmap f (Node a l r) = Node (f a) (fmap f l) (fmap f r)

但是,如果要确保二叉树上的映射保持平衡,则需要一个函数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
balanceTree :: Ord a => BinaryTree a -> BinaryTree a

您应该能够通过一些googling很容易地实现这个函数,然后您可以定义一个专门的映射函数。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
binMap :: (Ord a, Ord b) => (a -> b) -> BinaryTree a -> BinaryTree b
binMap f = balanceTree . fmap f

然后,您应该确保您和您的库的用户不会使用fmap (除非必要),而应该使用binMap

票数 4
EN

Stack Overflow用户

发布于 2014-04-06 13:01:09

如果要强制排序,则不能将二叉树作为函子,因为--正如您所指出的--类型不匹配。但是,虽然树不能是键上的函子,但它可以是值的函子,只要每个值都有单独的类型参数。标准Data.Map (也是作为搜索树实现的)是这样工作的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
-- Now the "v" parameter can be mapped over without any care for tree invariants
data Tree k v = Node k v (Tree k v) (Tree k v) | Leaf 

关于fmap的实现,您的第一个想法是正确的。不过,还有一种更懒散的方法,即让GHC派生实例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
{-# LANGUAGE DeriveFunctor #-}

data Tree k v = Node k v (Tree k v) (Tree k v) | Leaf deriving (Functor)

它几乎总是与您的意图相匹配,只需记住让最后一个类型参数是您想要映射的参数。

票数 5
EN

Stack Overflow用户

发布于 2014-04-06 18:34:07

我不打算说我推荐下面的内容,但是为了完整性,实际上可以定义这样一个函子。

Functor类型要求您可以将任何函数放入Functor中。这通常意味着很难确保需要类型实例的不变量。但是,我们可以在很大程度上控制这种情况,并最终得到一个Functor实例。实际上,这意味着我们可以使用类型系统来确保我们将我们的再平衡推迟到更方便的时间。

首先,我们将介绍对上述类型的需求。特别是,我们将给它一个维护平衡的Monoid实例。这很好,因为Monoid不要求容器是多态的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
instance Ord a => Monoid (BalancedTree a) where
  mempty = Leaf
  mappend Leaf Leaf = Leaf
  mappend Leaf b    = b
  mappend b    Leaf = b
  mappend (Node a l1 r1) (Node b l2 r2) = ... -- merge and rebalance here

现在,使用这个实例,我们可以编写几乎与Monad实例对应的BinaryTree函数。特别是,我们需要它来结合我们的新树作为构建使用bindBin,几乎版本的(>>=)上的二进制搜索树。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
returnBin :: a -> BinaryTree a
returnBin a = Node a Leaf Leaf

bindBin :: Ord b => BinaryTree a -> (a -> BinaryTree b) -> BinaryTree b
bindBin Leaf _ = Leaf
bindBin (Node a l r) f = bindBin l f <> f a <> bindBin r f

然后我们介绍这种非常奇怪的类型(它需要RankNTypes扩展)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
newtype FBinaryTree a = 
  FBinaryTree (forall r . Ord r => (a -> BinaryTree r) -> BinaryTree r)

考虑这个问题有很多种方法,但我们只需注意到,FBinaryTree aBinaryTree a之间有一个同构,基本上是由returnBinbindBin见证的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
toF :: BinaryTree a -> FBinaryTree a
toF bt = FBinaryTree (bindBin bt)

fromF :: Ord a => FBinaryTree a -> BinaryTree a
fromF (FBinaryTree k) = k returnBin

最后,由于FBinaryTree继承了Cont monad或Yoneda引理类型的一些属性,我们可以为FBinaryTree定义一个Functor实例!

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
instance Functor FBinaryTree where
  fmap f (FBinaryTree c) = FBinaryTree (\k -> c (k . f))

因此,现在我们所要做的就是将我们的BinaryTree转换成FBinaryTree,在那里执行Functor操作,然后根据需要返回到BinaryTree。一帆风顺,对吧?

好吧,差不多了。事实证明,我们为此付出了巨大的效率代价。特别是,在使用像FBinaryTree这样的类型时,很容易出现指数膨胀。我们可以通过不时地通过FBinaryTree通过BinaryTree发送

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
optimize :: Ord a => FBinaryTree a -> FBinaryTree a
optimize = toF . fromF

正如类型所示,它要求我们在那里有一个Ord实例。实际上,代码将在那里使用Ord实例来执行所需的再平衡。

票数 4
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22899700

复制
相关文章
Remove China Apps凉了?作死的阿三们并没有罢休
近日,一款名为『Remove China Apps』的应用,在印度火了,上架2周,下载量500W+,日均下载量超过20W,登上了Google PlayStore印度地区排行榜榜首,而TikTok(抖音国际版)仅位居第4。这款应用的功能,已经写得很明目张胆了——卸载中国应用。
Android扫地僧
2020/06/09
6020
正常的工作流程
修改文件,将它们更新的内容添加到索引中。 $ git add file1 file2 file3 你现在为commit做好了准备,你可以使用git diff命令再加上–cached参数,看看哪些文件将被提交(commit)。 (如果没有–cached参数,git diff会显示当前你所有已做的但没有加入到索引里的修改。)你也可以使用git status命令来获得当前项目的一个状况。
用户3004328
2018/09/06
7450
如何从Google Play下载Android应用的APK安装文件?
有时候可能因为种种原因,你无法直接在手机上连接Google Play来下载应用(比如说你设备不兼容,说你所在地区不支持,或者你想装到上不去Google Play的Kinlde上),但你又想安装这个应用,怎么办呢?
Enjoy233
2019/03/05
8.7K0
如何从Google Play下载Android应用的APK安装文件?
Anbox安装apk失败(提示Failure res=-113等)的解决方法
详细描述,如下(Anbox:如何安装Google Play商店并启用ARM(libhoudini)支持,简单方法):
zhangrelay
2019/01/31
8.4K0
Anbox安装apk失败(提示Failure res=-113等)的解决方法
uniapp下载apk并且安装(uniapp打包后apk白屏)
-alias xxx : xxx是别名 xxx.keystore : 文件名
全栈程序员站长
2022/08/01
6.5K0
uniapp下载apk并且安装(uniapp打包后apk白屏)
Apache编译后无法正常工作
因为某个场景的需求,要在一个国产系统Rocky4.2(国产凝思4.2操作系统)上安装Apache,虽说此系统是基于Redhat 5.8开发的,但是发现yum安装源包管理,RPM命令倒是能用,但是底层依赖完全没有,这就尴尬了,so,只能源码编译安装了。
后场技术
2020/09/03
2.8K0
网页离开时改变标题“崩溃欺骗”
我们先创建一个 js 文件,我们用记事本就好了,然后改个文件名,不妨就叫crash-cheat.js吧,你们可以随意! 然后把文件放到 source 文件夹的 js 文件夹的 src 里面。(我用的 next 主题,放这里统一存放,其他主题随意)
Cell
2022/02/25
1.2K0
缩小APK,增加下载量
原文地址:Shrinking APKs, growing installs: How your app’s APK size impacts install conversion rates 原文作者
Android 开发者
2018/05/31
2.9K0
iPhone Safari 下载企业包出现 apk
有人反馈企业包下载链接,使用 iPhone Safari 打开后出现下载 apk 的提示
莫空9081
2021/11/24
1.2K0
如何在程序崩溃时自动生成 stacktrace
有什么好的办法可以在 C/C++ 程序段错误退出时输出堆栈信息,来方便查找错误么?
ClearSeve
2022/02/10
1K0
pycharm调试教程_程序调试时应当用
在了解Python编程之前,我们需要先弄明白如何编写运行代码。所以非常有必要先讲解一下Python的集成开发环境,也就是IDE(Integrated Development Environment)。PyCharm是一款优秀的开源Python语言集成开发工具。PyCharm能够调试运行程序,另外它还提供了强大的代码提示功能。在PyCharm的下载页面能够指定安装系统选择付费版(Professional)或者免费版(Community)进行安装。付费版的PyCharm提供了更强大的Python服务器后端开发功能。这里我们以windows系统免费版(PyCharm Community)下载安装。我们只对PyCharm的基本功能进行简单概括,详细内容请查阅官方文档。PyCharm下载地址(https://www.jetbrains.com/PyCharm/download/#section=windows)
全栈程序员站长
2022/09/25
1.3K0
pycharm调试教程_程序调试时应当用
5分钟短文 | Android证书生成,签名,验证,虽然难,但学一次就够了!
从Android演进开始,APK签名就已经成为Android的一部分,并且android要求所有Apks都必须先签名,然后才能将其安装在设备上。关于如何生成密钥以及如何签名的文章很多。一个Apk,但我们将从安全角度进行研究。在对Apk文件进行反编译或反向工程之后,应查看哪个文件,以获取有关最初对应用进行签名的开发人员的更多信息。
程序员小助手
2020/06/17
1.1K0
直接下载google play应用-APK Downloader
作者:matrix 被围观: 4,603 次 发布时间:2013-11-08 分类:兼容并蓄 | 2 条评论 »
HHTjim 部落格
2022/09/26
4.8K0
直接下载google play应用-APK Downloader
如何在.NET程序崩溃时自动创建Dump?
首先能明确的一点是"程序崩溃退出了是不能用常规的方式 dump 的",因为整个进程树都已经退出。现场已经无法使用常规的方式读取到。
InCerry
2022/11/14
1.8K0
apk加壳加密工具(apk protect) v1.0下载「建议收藏」
apk加壳加密工具(apk_protect)是用于加密apk文件中dex文件的加密工具,加密的东西主要有字符串加密、流程加密、类名加密和api加密(未完成,后续支持)等,有于较好的保护apk文件,使之不易激活成功教程分析。__我对apk_protect在线加密的有效性进行了测试和分析,发现确实给android_apk提供了无法激活成功教程的加密壳。虽然在线加密已经是非常省时省力的了,但是仍然有不少程序员懒于折腾(尽管这已经不叫折腾了,就是上传一下再下载,比起写代码来说,这简直就是享受)。于是,意外的发现他们已经推出了懒人版apk_protect。没错,懒人版!也就是免安装单机版!无ads无插件无需安装,简单选定apk文件点击加密即可!_____使用方法___运行apkcrypt.exe,选择你所需要加密的apk,然后点击“add_apk_protect”。
全栈程序员站长
2022/09/14
1.7K0
win7怎么查看驱动是否正常工作
我们在使用电脑的时候经常会遇到各种各样的问题,今天我就教大家在电脑使用过程中出现问题时怎么检查电脑驱动是否正常齐全。
点云PCL博主
2019/07/30
2.5K0
VC调试时输出调试信息到Debug窗口
TRACE宏(afx.h, AfxTrace) (TRACE将信息输出到afxDump对象,只在_DEBUG定义时输出,最多输出512个字符,格式化与printf类似) afxDump对象(afx.h, CDumpContext) (afxDump调用OutputDebugString把信息输出到Debug窗口,继承CObject的类可以重载Dump方法格式化此类的Dump信息,输出时把afxDump作为Dump方法的参数) OutputDebugString(windows.h) (TRACE, afxDump在使用MFC时使用,不使用MFC时可以用OutputDebugString,AfxOutputDebugString和OutputDebugString用法一样)
战神伽罗
2019/07/24
1.7K0
点击加载更多

相似问题

从playstore下载时签名的apk崩溃

12

获取崩溃签名的apk /bundle,但正常调试apk工作正常

115

下载apk时从google playstore获取邮件id

20

从playstore下载时应用程序崩溃

00

如何在playstore ApK上安装调试APK

10
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文