programming-languages学习笔记–第9部分

*/-->

pre.src {background-color: #292b2e; color: #b2b2b2;}

pre.src {background-color: #292b2e; color: #b2b2b2;}

pre.src {background-color: #292b2e; color: #b2b2b2;}

pre.src {background-color: #292b2e; color: #b2b2b2;}

programming-languages学习笔记–第9部分

1 过程分解与面向对象分解

  • 函数式编程中,把程序分解为完成一些操作的函数。
  • 面向对象编程中,把程序分解为类,这些类为某些类型的数据提供行为。

这两种分解方式完全相反。
哪种方式更好看个人口味,但也依赖于你希望如何修改/扩展软件。对于包含两个或更多参数的操作,函数和模式匹配是很简明的,但是OOP可以使用double dispatch达到目的。

示例,实现一个表达式的小型语言:

  eval toString hasZero
Int        
Add        
Negate        
       

ML(函数式)中的标准方法:

  • 为每种变量(每一行)定义一个数据类型和一个构造器(在动态类型语言中,我们不会给数据类型一个名字,但是仍然按这样的方式考虑问题。)
  • 为每个操作定义一个函数
  • 以每列一个函数的方式填写表格,每个函数中针对每个单元格有一个分支;如果列中的多项相同,可以组合分支(使用通配模式)。

这个方法就是过程分解:对问题分解为每个操作有一个对应的过程。

datatype exp =
Int of int
| Negate of exp
| Add of exp * exp exception BadResult of string fun add_values (v1, v2) =
case (v1, v2) of
(Int i, Int j) => Int (i+j)
| _ => raise BadResult "non-ints in addition" fun eval e =
case e of
Int _ => e
| Negate e1 => (case eval e1 of
Int i => Int (~i)
| _ => raise BadResult "non-int in negation")
| Add(e1, e2) => add_values(eval e1, eval e2) fun toString e =
case e of
Int i => Int.toString i
| Negate e1 => "-(" ^ (toString e1) ^ ")"
| Add(e1,e2) => "(" ^ (toString e1) ^ ")" ^ " + "
^ (toString e2) ^ ")" fun hasZero e =
case e of
Int i => i=
| Negate e1 => hasZero e1
| Add(e1, e2) => (hasZero e1) orelse (hasZero e2)

OO的标准方法:

  • 为表达式定义一个类,为每个操作(每一列)定义一个抽象方法,(ruby中不需要,动态类型中不需要指定抽象方法)
  • 为每个数据变量(每一行)定义一个子类
  • 在每个子类中,为每个操作定义一个方法。

这个方法是面向数据的分解:把问题分解为每个数据变量对应一个类。

class Exp
# 可以在这里写默认实现或辅助函数
end class Value < Exp
end class Int < Value
attr_reader :i
def initialize i
@i = i
end def eval
self
end def toString
@i.to_s
end def hasZero
@i ==
end
end class Add < Exp
attr_reader :e1, :e2
def initialize(e1, e2)
@e1 = e1
@e2 = e2
end def eval
Int.new(@e1.eval.i + @e2.eval.i)
end def toString
"(" + @e1.toString + " + " + @e2.toString + ")"
end def hasZero
@e1.hasZero || @e2.hasZero
end
end class Negate < Exp
attr_reader :e
def initialize(e)
@e = e
end def eval
Int.new(-@e.eval.i)
end def toString
"-(" + @e.toString + ")"
end def hasZero
@e.hasZero
end
end

总结:

  • FP和OOP总是按照相反的方式做同样的事,按行或按列组织程序
  • 哪个更自然取决你做什么(解释器或GUI)或者个人爱好
  • 代码布局是重要的,但是没有完美的方法,因为软件有许多种结构维度。工具,IDE可以给你多种视图(行/列)

面向对象首先关心的是对象,然后是针对这些对象有哪些操作。
函数式首先关心的是操作,然后是这些操作针对哪些数据。

2 扩展代码:添加操作或变体(variant)

可扩展性,扩展上一节的程序:

  eval toString hasZero noNegConstants
Int        
Add        
Negate        
Mult        

函数式方式:

  • 容易添加新的操作(增加一个函数),比如noNegConstants,不需要修改以前的代码
  • 添加新的变体需要修改旧函数,但是ML类型检查会给出todo list(在原先的代码没有通用匹配的情况下),

OOP方式:

  • 容易添加新的变体(增加一个类即可)
  • 添加一个新的操作需要修改旧的类,但是Java的类型检查会在原先的代码没有默认方法的情况下给出todo list

不管是函数式还是OOP,都可以提前计划方便以后新增变体或操作。
函数式用High-order function添加新类型, OOP使用双派发(double-dispatch)模式添加新操作。

未来是难以预测的,我们或许不知道需要什么样的扩展,或者两种扩展都需要。
可扩展性是一把双刃剑:

  • 以后不需要修改就可以代码重用
  • 但需要写更多的原始代码
  • 原始代码更难理解或修改
  • 一些语言机制让代码更难扩展,比如ML的模块隐藏了数据类型;Java的final阻止子类化/覆盖。

3 使用函数式分解二元方法

使Add支持更多操作:

  Int String Rational
Int      
String      
Rational      

函数式中使用case表达式就解决了,因为函数首先关心的就是操作,针对不同数据之间的操作用case表达式。

datatype exp = Add of exp * exp
| Int of int
| Negate of exp
| String of string
| Rational of int * int fun add_values (v1, v2) =
case (v1, v2) of
(Int i, Int j) => Int (i+j)
| (Int i, String s) => String(Int.toString i ^ s)
| (Int i, Rational(j,k)) => Rational(i*k+j,k)
| (String s, Int i) => String(s ^ Int.toString i)
| (String s1, String s2) => String(s1 ^ s2)
| (String s, Rational(i,j)) => String(s ^ Int.toString i ^ "/"
^ Int.toString j)
| (Rational _, Int _) => add_values(v2, v1)
| (Rational(i, j), String s) => String(Int.toString i ^ "/"
^ Int.toString j ^ s)
| (Rational(a,b), Rational(c,d)) => Rational(a*d+b*c, b*d)
| _ => raise BadResult "non-values passed to add_values"

4 双重派发

OOP更关心对象,一个消息发送过来,不知道怎么处理,让对象自己去处理,就是双重派发,也可以做到三重、四重(一个操作作用于三个、四个对象),但要写很多方法。

class Add < Exp
def eval
e1.eval.add_values e2.eval
end
end class Int < Value
# OOP中的双重派发
def add_values v
v.addInt self
end
def addInt v
Int.new(v.i + i)
end
end

5 Multimethods

也叫多重派发。
针对不同对象的同一操作,用同一个方法名,自动调用对应的方法。

ruby动态类型,方法不能重名,因此没有多重方法。Java和C++是静态类型,一个类可以有重名方法,但在编译时确定了参数的类型,叫做静态重载。

许多OOP语言有multimethods。比如Clojure中的multimethod和Scala中的trait。

6 多重继承

单继承的类层次是一个树,多重继承的类层次更复杂。

7 Mixins

mixin是一个方法集合,没有实例。含有mixins的语言的类大都只能有一个父类,但可以包含多个mixin.

module Doubler
def double
self + self
end
end class Pt
attr_accessor :x, :y
include Doubler
def + other
ans = Pt.new
ans.x = self.x + other.x
ans.y = self.y + other.y
ans
end
end class String
include Doubler
end

Ruby中最大的两个mixins是Comparable和Enumerable。

class MyRange
include Enumerable
def initialize(low, high)
@low = low
@high = high
end # 支持low>high的情况
def each
if @low <= @high
i=@low
while i <= @high
yield i
i=i+
end
else
i=@low
while i >= @high
yield i
i=i-
end
end
end
end for i in MyRange.new(,)
for j in MyRange.new(,)
print (i+j).to_s + " "
end
puts
end MyRange.new(,).each { |x|
MyRange.new(,).each { |y|
print (x+y).to_s + " " }
puts }

8 接口

静态类型中的类是一个类型。
接口也是一个类型,但不是一个类.

9 抽象方法

静态类型中支持覆盖的方法就是抽象方法。

作者: ntestoc

Created: 2019-01-06 日 22:19

programming-languages学习笔记--第9部分的更多相关文章

  1. CUDA Programming Guide 学习笔记

    CUDA学习笔记 GPU架构 GPU围绕流式多处理器(SM)的可扩展阵列搭建,每个GPU有多个SM,每个SM支持数百个线程并发执行.目前Nvidia推出了6种GPU架构(按时间顺序,详见下图):Fer ...

  2. Programming Erlang 学习笔记(一)

    入门 启动Shell 在cmd中输入命令”erl”,百分号(%)表示一个注释的开始,从百分号开始到这行结束的所有文本都被看做是注释. 一个完整的命令需要以一个句点和一个回车结束. 退出erlang的命 ...

  3. UIView Programming Guide学习笔记

    |View |Creating and Configuring View Objects |Creating and Managing a View Hierarchy |Adjusting the ...

  4. The C++ Programming Language 学习笔记 第7章 函数

    1.关于内联函数(inline)      借用一下书中的例子. inline int fac(int n) { ) ? :n*fac(n-); }      inline描述符给编译器一个提示,要求 ...

  5. The C++ Programming Language 学习笔记 第6章 表达式和语句

    1.关于strcpy函数. 书中说c风格的字符串尽量少用,strcpy这样的函数应该也要少用.这里讲这个函数主要是要通过本章课后练习第十题来讲一下前面提及的要点.巩固一下前几章的知识.写了一段,本来感 ...

  6. The C++ Programming Language 学习笔记 第5章 指针、数组和结构

    1.关于输出指向字符的指针的值. 现在定义,char c='a',char* pc=&c.在C中,输出该值只需要printf("%p\n",pc);而在C++中,如果cou ...

  7. The C++ Programming Language 学习笔记 第四章 类型和声明

    1.关于main 函数中的 return 0 C99标准中,main 函数的返回值类型必须是 int ,这样返回值才能传递给程序的激活者(如操作系统).如果 main 函数的最后没有写 return ...

  8. 3D Game Programming withDX11 学习笔记(一) 数学知识总结

    在图形学中,数学是不可或缺的一部分,所以本书最开始的部分就是数学知识的复习.在图形学中,最常用的是矢量和矩阵,所以我根据前面三个章节的数学知识,总结一下数学知识. 一.矢量 数学中的矢量,拥有方向和长 ...

  9. Learning ROS for Robotics Programming Second Edition学习笔记(十) indigo Gazebo rviz slam navigation

    中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 moveit是书的最后一章,由于对机械臂完全不知,看不懂 ...

  10. Learning ROS forRobotics Programming Second Edition学习笔记(八)indigo rviz gazebo

    中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS forRobotics Pro ...

随机推荐

  1. WIN2008服务器不能复制粘贴怎么办

    在任务管理器界面,选择进程,找到rdpclip.exe进程.   找到该进程后,点击结束进程.   然后点击WINDOWS任务管理器左上角的[文件]—[新建任务].   在编辑框内输入rdpclip. ...

  2. 设计模式学习——代理模式(Proxy Pattern)之 强制代理(强校验,防绕过)

    上周温习了代理模式:http://www.cnblogs.com/chinxi/p/7354779.html 在此进行拓展,学习强制代理.但是发现网上大多例子都有个“天坑”(我是这么认为的),在得到代 ...

  3. [js高手之路]Node.js实现简易的爬虫-抓取博客所有文章列表信息

    抓取目标:就是我自己的博客:http://www.cnblogs.com/ghostwu/ 需要实现的功能: 抓取博客所有的文章标题,超链接,文章摘要,发布时间 需要用到的库: node.js自带的h ...

  4. AngularJs 第一个自定义指令编写

    公司在做一个OA系统, 包括移动端(从微信企业号进入OA系统),电脑端. 电脑端还是用的传统的easyui做界面,asp.net mvc作为服务端.这个技术已经很成熟了配合权限框架很快就能开发出来.但 ...

  5. 【Python】Java程序员学习Python(八)— 基本类型的基本运算

    这一篇待写,毕竟基本运算都是通用的.

  6. Automate the Sizing of your SGA in Oracle 10g

    How much memory does each of the individual components of the SGA need? Oracle now has methods to de ...

  7. redis介绍(7)高级用法

    redis的过期策略以及内存淘汰机制 分析:这个问题其实相当重要,到底redis有没用到家,这个问题就可以看出来.比如你redis只能存5G数据,可是你写了10G,那会删5G的数据.怎么删的,这个问题 ...

  8. 转:socket

    最近浏览了几篇有关Socket发送消息的文章,发现大家对Socket Send方法理解有所偏差,现将自己在开发过程中对Socket的领悟写出来,以供大家参考. (一)架构 基于TCP协议的Socket ...

  9. [翻译] SIAlertView

    SIAlertView https://github.com/Sumi-Interactive/SIAlertView An UIAlertView replacement with block sy ...

  10. 对MBProgressHUD进行二次封装并精简使用

    对MBProgressHUD进行二次封装并精简使用 https://github.com/jdg/MBProgressHUD 几个效果图: 以下源码是MBProgressHUD支持最新的iOS8的版本 ...