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 条评论
登录 后参与评论

相关文章

来自专栏Golang语言社区

【Go 语言社区】Go学习笔记:json处理

Encode 将一个对象编码成JSON数据,接受一个interface{}对象,返回[]byte和error: func Marshal(v interfac...

67912
来自专栏技术博客

C#简单的面试题目(二)

实例化对象  new Class();        隐藏基类方法,即覆盖方法  public  new xxx(){}

1281
来自专栏开发与安全

从零开始学C++之模板(二):类模板、Stack的类模板实现(自定义链栈方式,自定义数组方式)

一、类模板 类模板:将类定义中的数据类型参数化 类模板实际上是函数模板的推广,可以用相同的类模板来组建任意类型的对象集合 (一)、类模板的定义 templ...

2320
来自专栏逸鹏说道

Python3 与 C# 基础语法对比(List、Tuple、Dict专栏)

Python3 与 C# 基础语法对比(基础知识场):https://www.cnblogs.com/dotnetcrazy/p/9102030.html

24610
来自专栏Golang语言社区

在Go语言中使用JSON

Encode 将一个对象编码成JSON数据,接受一个interface{}对象,返回[]byte和error: func Marshal(v interfac...

3519
来自专栏极乐技术社区

使用ES6新特性开发微信小程序(1)

ECMAScript 6(简称ES6)是JavaScript语言的最新标准。因为当前版本的ES6是在2015年发布的,所以又称ECMAScript 2015。 ...

2255
来自专栏desperate633

LintCode 单词切分题目分析

给出一个字符串s和一个词典,判断字符串s是否可以被空格切分成一个或多个出现在字典中的单词。

972
来自专栏Ryan Miao

String.split()用法以及特殊分隔符注意,ps:|

转载:http://www.cnblogs.com/mingforyou/archive/2013/09/03/3299569.html 在java.lang包...

3119
来自专栏飞雪无情的博客

Go语言参数传递是传值还是传引用

其实对于传值和传引用,是一个比较古老的话题,做研发的都有这个概念,但是可能不是非常清楚。对于我们做Go语言开发的来说,也想知道到底是什么传递。

2333
来自专栏服务端技术杂谈

Java编码规范

命名 类名使用UpperCamelCase风格。 领域模型相关命名:DO / DTO / VO / DAO等。 方法名,参数名,成员变量,局部变量都统一使用lo...

3314

扫码关注云+社区

领取腾讯云代金券