Monoid是数学范畴理论(category theory)中的一个特殊范畴(category)。不过我并没有打算花时间从范畴理论的角度去介绍Monoid,而是希望从一个程序员的角度去分析Monoid以及它在泛函编程里的作用。从这个思路出发我们很自然得出Monoid就是一种数据类型,或者是一种在泛函编程过程中经常会遇到的数据类型:当我们针对List或者loop进行一个数值的积累操作时我们就会使用到Monoid。实际上Monoid就是List[A] => A的抽象模型。好了,我们就不要越描越黑了吧,还是看看Monoid的定义吧:

Monoid由以下条件组成:

1、一个抽象类型A

2、一个二元结合性函数(binary associative function),对传入的两个A类参数进行操作后产生一个A类型结果

3、一个恒等值(identity)

由于Monoid是一个数学类型,它的二元操作函数必须遵循一些定律:

1、结合性(associativity):op(a,op(b,c)) = op(op(a,b),c):这个定律是函数组合(function composition)不可缺的条件

2、二元函数参数中如果有一个是恒等值时操作结果为另一个参数:op(identity,v) = v

我们可以用编程语言来描述Monoid:

   trait Monoid[A] {                //被封装的类型A
def op(a1: A, a2: A): A //二元函数
val zero: A //恒等值identity
}

我们用scala的特质(trait)描述了Monoid。它就是一个抽象的数据类型。

既然Monoid trait是个抽象类型,那么我们可以试着创建几个基础类型的Monoid实例:

   val stringConcatMonoid = new Monoid[String] {
def op(s1: String, s2: String) = s1 + s2
val zero = "" // op(zero,s2) = "" + s2 = s2 恒等值定律
} //> stringConcatMonoid : ch10.ex1.Monoid[String] = ch10.ex1$$anonfun$main$1$$an
//| on$1@3581c5f3
val intAdditionMonoid = new Monoid[Int] {
def op(i1: Int, i2: Int) = i1 + i2
val zero = 0
} //> intAdditionMonoid : ch10.ex1.Monoid[Int] = ch10.ex1$$anonfun$main$1$$anon$4
//| @340f438e
val intMultiplicationMonoid = new Monoid[Int] {
def op(i1: Int, i2: Int) = i1 * i2
val zero = 1
} //> intMultiplicationMonoid : ch10.ex1.Monoid[Int] = ch10.ex1$$anonfun$main$1$$
//| anon$5@30c7da1e

可以看出,这几个Monoid实例都符合Monoid定律。那我们可以先试着用用。上面提到Monoid最适合一串值的累加操作List[A] => A,我们可以对List[A]进行操作示范:

  def reduce[A](as: List[A])(m: Monoid[A]): A = {
as match {
case Nil => m.zero
case h::t => m.op(h, reduce(t)(m))
}
} //> reduce: [A](as: List[A])(m: ch10.ex1.Monoid[A])A

Monoid m是个抽象类型,m.zero和m.op()的具体意义要看Monoid的实例了:

   reduce(List(1,2,3))(intAdditionMonoid)          //> res3: Int = 6
reduce(List("this is ","the string", " monoid"))(stringConcatMonoid)
//> res4: String = this is the string monoid

对List[A]的具体累加处理是按照intAdditionMonoid和stringConcatMonoid的二元函数功能进行的。看来Monoid特别适用于List类型的循环操作。可以把reduce函数的参数拓展开来看看:

   reduce[A](as: List[A])(zero: A)(op: (A,A) => A) : A

这个类型款式跟折叠算法的类型款式非常相似:

   def foldRight[A,B](as: List[A])(z: B)(f: (A,B) => B): B
如果类型B=类型A
def foldRight[A](as: List[A])(z: A)(f: (A,A) => A): A

实际上我们可以直接用上面的Monoid实例运算折叠算法:

   List(1,2,3).foldRight(intAdditionMonoid.zero)(intAdditionMonoid.op)
//> res3: Int = 6
List("this is ","the string", " monoid").foldLeft(stringConcatMonoid.zero)(stringConcatMonoid.op)
//> res4: String = this is the string monoid

左右折叠算法都可以。Monoid的结合性定律(associativity law)可以使List元素运算左右路径相等。

下面我们再试着增加几个Monoid实例:

   def optionMonoid[A] = new Monoid[Option[A]] {
def op(o1: Option[A], o2: Option[A]): Option[A] = o1 orElse o2
val zero = None // op(zero, o1)= None orElse o2 = o2
} //> optionMonoid: [A]=> ch10.ex1.Monoid[Option[A]]{val zero: None.type}
def listConcatMonoid[A] = new Monoid[List[A]] {
def op(l1: List[A], l2: List[A]) = l1 ++ l2
val zero = Nil
} //> listConcatMonoid: [A]=> ch10.ex1.Monoid[List[A]]{val zero: scala.collection.
//| immutable.Nil.type}
val booleanOrMonoid = new Monoid[Boolean] {
def op(b1: Boolean, b2: Boolean) = b1 || b2
val zero = false
} //> booleanOrMonoid : ch10.ex1.Monoid[Boolean] = ch10.ex1$$anonfun$main$1$$anon
//| $6@5b464ce8
val booleanAndMonoid = new Monoid[Boolean] {
def op(b1: Boolean, b2: Boolean) = b1 && b2
val zero = true
} //> booleanAndMonoid : ch10.ex1.Monoid[Boolean] = ch10.ex1$$anonfun$main$1$$an
//| on$7@57829d67
def endoComposeMonoid[A] = new Monoid[A => A] {
def op(f: A => A, g: A => A) = f compose g
val zero = (a: A) => a // op(zero, g: A => A) = zero compose g = g
} //> endoComposeMonoid: [A]=> ch10.ex1.Monoid[A => A]
def endoAndThenMonoid[A] = new Monoid[A => A] {
def op(f: A => A, g: A => A) = f andThen g
val zero = (a: A) => a // op(zero, g: A => A) = zero andThen g = g
} //> endoAndThenMonoid: [A]=> ch10.ex1.Monoid[A => A]
//计算m的镜像Monoid
def dual[A](m: Monoid[A]) = new Monoid[A] {
def op(x: A, y: A) = m.op(y,x) //镜像op即时二元参数位置互换
val zero = m.zero
} //> dual: [A](m: ch10.ex1.Monoid[A])ch10.ex1.Monoid[A]
def firstOfDualOptionMonoid[A] = optionMonoid[A]
//> firstOfDualOptionMonoid: [A]=> ch10.ex1.Monoid[Option[A]]{val zero: None.ty
//| pe}
def secondOfDualOptionMonoid[A] = dual(firstOfDualOptionMonoid[A])
//> secondOfDualOptionMonoid: [A]=> ch10.ex1.Monoid[Option[A]]

以上几个增加的Monoid实例中endoComposeMonoid和endoAndThenMonoid可能比较陌生。它们是针对函数组合的Monoid。

还是回到对List[A]的累加操作。下面这个函数用Monoid对List[A]元素A进行累加操作:

   def concatenate[A](l: List[A], m: Monoid[A]): A = {
l.foldRight(m.zero){(a,b) => m.op(a,b)}
} //> concatenate: [A](l: List[A], m: ch10.ex1.Monoid[A])A
concatenate[Int](List(1,2,3),intAdditionMonoid) //> res0: Int = 6

那么如果没有List[A]元素A类型Monoid实例怎么办?我们可以加一个函数:

 def foldMap[A,B](as: List[A])(m: Monoid[B])(f: A => B): B

如果我们有一个函数可以把A类转成B类 A => B,那我们就可以使用Monoid[B]了:

   def foldMap[A,B](as: List[A])(m: Monoid[B])(f: A => B): B = {
as.foldRight(m.zero)((a,b) => m.op(f(a),b))
}

说明一下:foldRight的类型款式:foldRight[A,B](as: List[A])(z: B)(g: (A,B) => B): B。其中(A,B) => B >>> (f(A),B) => B >>> (B,B) => B 就可以使用 Monoid[B].op(B,B)=B了。我们也可以用foldLeft来实现foldMap。实际上我们同样可以用foldMap来实现foldRight和foldLeft:

 def foldRight[A,B](la: List[A])(z: B)(f: (A,B) => B): B
def foldLeft[A,B](la: List[A])(z: B)(f: (A,B) => B): B
def foldMap[A,B](as: List[A])(m: Monoid[B])(f: A => B): B

foldRight和foldLeft的f函数是(A,B) => B,如果用curry表达:A => (B => B),如果能把 A => ? 转成 B => B,那么我们就可以使用endoComposeMonoid[B].op(f: B => B, g: B => B): B。

   def foldRight[A,B](as: List[A])(z: B)(f: (A,B) => B): B = {
foldMap(as)(endoComposeMonoid[B])(a => b => f(a,b))(z)
}

说明:foldMap需要f: A => B, foldRight有 (A,B) => B >>> A => B => B >>> f(a)(b) => b >>> f(a,b)(z) >>> f(b)(b)

foldLeft是从左边开始折叠,只需要采用endoComposeMonoid的镜像Monoid把op参数位置调换就行了:

   def foldLeft[A,B](as: List[A])(z: B)(f: (A,B) => B): B = {
foldMap(as)(dual(endoComposeMonoid[B]))(a => b => f(a,b))(z)
}

在这节我们简单的介绍了Monoid及它的一些初级类型的实例使用方式。我们也把Monoid代数模型的一面:函数的互通转换及组合稍微示范了一下。在下一节我们将会把Monoid在实际编程中的应用以及Monoid的深度抽象做些讨论。

泛函编程(21)-泛函数据类型-Monoid的更多相关文章

  1. 泛函编程(5)-数据结构(Functional Data Structures)

    编程即是编制对数据进行运算的过程.特殊的运算必须用特定的数据结构来支持有效运算.如果没有数据结构的支持,我们就只能为每条数据申明一个内存地址了,然后使用这些地址来操作这些数据,也就是我们熟悉的申明变量 ...

  2. 泛函编程(24)-泛函数据类型-Monad, monadic programming

    在上一节我们介绍了Monad.我们知道Monad是一个高度概括的抽象模型.好像创造Monad的目的是为了抽取各种数据类型的共性组件函数汇集成一套组件库从而避免重复编码.这些能对什么是Monad提供一个 ...

  3. 泛函编程(23)-泛函数据类型-Monad

    简单来说:Monad就是泛函编程中最概括通用的数据模型(高阶数据类型).它不但涵盖了所有基础类型(primitive types)的泛函行为及操作,而且任何高阶类或者自定义类一旦具备Monad特性就可 ...

  4. 泛函编程(25)-泛函数据类型-Monad-Applicative

    上两期我们讨论了Monad.我们说Monad是个最有概括性(抽象性)的泛函数据类型,它可以覆盖绝大多数数据类型.任何数据类型只要能实现flatMap+unit这组Monad最基本组件函数就可以变成Mo ...

  5. 备份-泛函编程(23)-泛函数据类型-Monad

    泛函编程(23)-泛函数据类型-Monad http://www.cnblogs.com/tiger-xc/p/4461807.html https://blog.csdn.net/samsai100 ...

  6. 泛函编程(27)-泛函编程模式-Monad Transformer

    经过了一段时间的学习,我们了解了一系列泛函数据类型.我们知道,在所有编程语言中,数据类型是支持软件编程的基础.同样,泛函数据类型Foldable,Monoid,Functor,Applicative, ...

  7. 泛函编程(36)-泛函Stream IO:IO数据源-IO Source & Sink

    上期我们讨论了IO处理过程:Process[I,O].我们说Process就像电视信号盒子一样有输入端和输出端两头.Process之间可以用一个Process的输出端与另一个Process的输入端连接 ...

  8. 泛函编程(34)-泛函变量:处理状态转变-ST Monad

    泛函编程的核心模式就是函数组合(compositionality).实现函数组合的必要条件之一就是参与组合的各方程序都必须是纯代码的(pure code).所谓纯代码就是程序中的所有表达式都必须是Re ...

  9. 泛函编程(32)-泛函IO:IO Monad

    由于泛函编程非常重视函数组合(function composition),任何带有副作用(side effect)的函数都无法实现函数组合,所以必须把包含外界影响(effectful)副作用不纯代码( ...

随机推荐

  1. 阿里前DBA的故事

    别人怎么享受生活,与你无关.你怎么磨砺与你有头.引用同事周黄江的一句话,很多人努力程度还远没有到拼天赋的时候. 成功的人都是那种目标很明确的人.对于文中厨师的经历很感兴趣.不管是IT还是餐饮,哪个行业 ...

  2. 不同iOS版本做代码适配__IPHONE_OS_VERSION_MAX_ALLOWED 和 __IPHONE_8_0等专业术语

    目前开发只想最低版本支持iOS8了,iOS8以前的就不管了,然后现在iOS9和iOS10出来以后,有些新的API,也有些弃用的API,为了兼容,有时候代码里面需要编写判断不同iOS版本,或者只允许指定 ...

  3. 一致性哈希算法 - consistent hashing

    1 基本场景比如你有 N 个 cache 服务器(后面简称 cache ),那么如何将一个对象 object 映射到 N 个 cache 上呢,你很可能会采用类似下面的通用方法计算 object 的 ...

  4. 开发Android必知的工具

    程序开发有时候非常依赖使用的开发工具,好的完备的开发工具可以让开发人员的工作效率有大幅度的提高.开发Android也是如此,大家可能都离不开Eclipse或Android Studio这些工具,但他们 ...

  5. 关于windows的service编程

    最近需要学习下windows的service编程框架,查了下msdn发现不知所云.于是谷歌之,发现了一个非常不错的文章,重点推荐讲的非常详细,深入,看完之后基本上就能很清楚windows的servic ...

  6. Java NIO原理分析

    Java IO 在Client/Server模型中,Server往往需要同时处理大量来自Client的访问请求,因此Server端需采用支持高并发访问的架构.一种简单而又直接的解决方案是“one-th ...

  7. 天猫浏览型应用的CDN静态化架构演变

    原文链接:http://www.csdn.net/article/2014-01-22/2818227-CDN-Architecture 在天猫双11活动中,商品详情.店铺等浏览型系统,通常会承受超出 ...

  8. Struts2知多少(2) Struts2 是什么

    Struts2是流行和成熟的基于MVC设计模式的Web应用程序框架. Struts2不只是Struts1下一个版本,它是一个完全重写的Struts架构. WebWork框架开始以Struts框架为基础 ...

  9. [IR] Link Analysis

    网络信息的特点在于: Query: "IBM" --> "Computer" --> documentIDs. In degree i 正比于 1/ ...

  10. Unity 3D 中自动寻路 和 跟随转向 探析

    这里主要讲三个函数 , 一个自动跟随函数 和 两个指向旋转函数 , 这三个函数在游戏角色创建过程中会经常用到: 这个是跟随函数 和 欧拉角旋转函数 public class GensuiZhixian ...