Scala typeclass 设计模式

前言

本文的写作的灵感主要是看了这个视频 : Tutorial: Typeclasses in Scala with Dan Rosen

加上查阅了相关的资料,觉得可以写一篇博客,再加上也很久没写博客了。本文的主要内容

是根据参考资料对typeclass的解释再加上自己的一点点理解,代码会借(cao)鉴(xi)资料

中的例子(不过代码会稍作修改)。

正文

typeclass定义

首先简单看看维基上对于typeclass的定义:"In computer science, a type class is a type

system construct that supports ad hoc polymorphism." 这个“ad hoc polymorphism”

(特质多态)其实也被称作函数重载或运算符重载。

在scala中采用typeclass模式有什么有优点呢?总的来说就是:代码易扩展;代码写得好看。

在例子中理解 typeclass

以下用到的代码均借鉴自[2] 。

首先我们来看看两个 ADT 的定义:

就是定义了一个表达式ADT,还有Json ADT。

然后还有给表达式赋值的和输出Json的两个object :

简单测试一下:

然后现在想添加一个功能,就是给定某个类型的对象,获得该对象的json字符串。面向对象

的做法是声明一个JsonConvertible接口,然后让有需要类型去继承该接口,实现 convertToJson

方法。

现在想让Expression类型的对象都能转成Json类型,那么就是每个Expression类型都要去实现

converToJson方法。

然后我们可以简单测试一下:

然后现在问题来了,我们想保持Expression接口尽量的轻量,不想有太多的依赖,而且如果现在

又要增加一个新的trait那么Number、Multiply和Divide三个类又要去实现这个接口的方法。或者

如果Expression是属于第三方的库,无法修改来继承JsonConvertible怎么办?

这时候继承多态不适用了,我们要用特质多态来解决这个问题。

我们先来看看新的write函数定义,现在我们新加了一个helper类JsonConverter,这个类实现了

把value转化为Json类型的方法。然后我们来看看现在这么做的好处:

我们可以看到,现在Expression可以保持原来的样子,并且把Expression转化为Json的逻辑

与Expression的实现分开了。简单测试一下:

so far so good。但是现在想再进一步使实现更简洁一些,这个JsonConverter其实并不一定要

显式的传入,我们可以借助scala的implicit来实现。

就是在原来的基础上作些小修改,把expressionJsonConverter改为implicit,还有write函数

改为curry,conv参数改为implicit。简单测试一下:

更进一步我们可以用scala中的context bound来改写代码。

context bound 的表达形式是 A : B,意思是在上下文中存在隐式的 B[A] 类型的对象。

刚开始接触的时候我觉得context bound 和 view bound很像, A <% B,view bound的意思

你可以把A当作B来用,上下文中需要存在一个A到B的隐式转换。

ok,到此就是scala type class的简单介绍,视频中后面还有关于Expression和Json的重构并

扩展到 Int 和 Tuple 上,有兴趣的可以看看。

相关资料

[1] What are type classes in Scala useful for?

[2] Tutorial: Typeclasses in Scala with Dan Rosen

[3] The Neophyte's Guide to Scala Part 12: Type Classes

[4] Type class

原创声明,本文系作者授权云+社区发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Ldpe2G的个人博客

Scala typeclass 设计模式

1006
来自专栏破晓之歌

JAVA入门3-2(未完,待续) 原

List(序列)、Queue(队列)可重复排列有序的,Set(集)不可重复无序。list和set常用。

905
来自专栏Java技术栈

IntegerCache的妙用和陷阱!

考虑下面的小程序,你认为会输出为什么结果? public class Test { public static void main(String[...

3345
来自专栏Java技术

Java 虚拟机内存区域划分详解(1)

JVM,java virtual machine, 即Java虚拟机,是运行java class文件的程序。

714
来自专栏葡萄城控件技术团队

深入浅出OOP(一): 多态和继承(早期绑定/编译时多态)

在本系列中,我们以CodeProject上比较火的OOP系列博客为主,进行OOP深入浅出展现。 无论作为软件设计的高手、或者菜鸟,对于架构设计而言,均需要多次重...

1936
来自专栏Vamei实验室

Python进阶08 异常处理

异常处理 在项目开发中,异常处理是不可或缺的。异常处理帮助人们debug,通过更加丰富的信息,让人们更容易找到bug的所在。异常处理还可以提高程序的容错性。 我...

1758
来自专栏码匠的流水账

聊聊GenericObjectPool的泄露检测

本文主要聊聊GenericObjectPool的abandon参数。主要用来做连接池的泄露检测用。

461
来自专栏顶级程序员

全面深入理解Python面向对象编程

面向过程编程最易被初学者接受,其往往用一长段代码来实现指定功能,开发过程中最常见的操作就是粘贴复制,即:将之前实现的代码块复制到现需功能处。

775
来自专栏编程

Java反射札记

Java反射相关内容,在阅读一些开源框架和自己动手封装逻辑时常要用到,以前陆陆续续地看到过一些文章,但是最终留下的是破碎的代码片段和类名,所以这次重新敲一遍反射...

18210
来自专栏walterlv - 吕毅的博客

使用 ExceptionDispatchInfo 捕捉并重新抛出异常

发布于 2017-10-23 14:22 更新于 2017-10...

411

扫码关注云+社区