博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Scala:函数式对象的定义及其使用
阅读量:6480 次
发布时间:2019-06-23

本文共 3525 字,大约阅读时间需要 11 分钟。

 Scala 中,类(class)的定义比 Java 面相对象的类定义要简洁一些,因为 Scala 在类定义时就同时定义了该类的主构造器(primary constructor。我们知道,构造函数只不过是一种特殊的函数(或称为方法)而已,所以 Scala 中的类定义个人觉得跟方法、函数定义的格式看起来差不多。

 

所谓的函数式对象,其实就是指对象内部没有可变的状态,即类里面的字段都是用关键字 valvalue的意思)定义的。

 

借助《Programming in Scala》中的例子,我们定义一个有理数(Rational)类,在此我们将有理数表示为分数的形式,即1 / 23 / 5的格式。其要求如下:

1  该类的对象(即任意有理数)能够相互之间进行一些基本运算,如加、减、乘、除,此外,还应该能与正数进行这些运算;

2  确保构造出有理数(Rationla 对象)时不能出现分母为零的情况;

3  重写 toString()方法,以最简分数的格式打印出 Rational 对象;

4  可以只给定一个分子来构造有理数,此时我们默认分母为 1,于是需要重载(overload)构造函数,Scala 中称为从构造器

 

看一下实现代码:

 

1.            // 有理数类  

2.            class Rational(n: Int, d: Int){  

3.             

4.                //在构造对象时确保分母不为零  

5.                require(d != 0)  

6.             

7.                // 获得分子分母的最大公约数  

8.                private val g = gcd(n.abs, d.abs)  

9.                val numer: Int = n / g  // 分子  

10.            val denom: Int = d / g  // 分母  

11.         

12.            // 重载构造器,此时的有理数实为整数  

13.            def this(n: Int) = this(n, 1)  

14.         

15.            // 重写toString,必须加上override关键字  

16.            override def toString = {  

17.                if(denom == 1)  

18.                    numer.toString  

19.                else 

20.                    numer + "/" + denom  

21.            }  

22.         

23.            // 常见的add方法  

24.            def add(that: Rational) :Rational = {  

25.                new Rational(  

26.                    numer * that.denom + that.numer * denom,  

27.                    denom * that.denom  

28.                )  

29.            }  

30.              

31.            // 操作符形式的加法  

32.            def +(that: Rational): Rational ={  

33.                new Rational(  

34.                    numer * that.denom + that.numer * denom,  

35.                    denom * that.denom  

36.                )  

37.            }  

38.         

39.            def *(that: Rational): Rational ={  

40.                new Rational(  

41.                    numer * that.numer,  

42.                    denom * that.denom  

43.                )  

44.            }  

45.         

46.            def -(that: Rational): Rational ={  

47.                new Rational(  

48.                    numer * that.denom - that.numer * denom,  

49.                    denom * that.denom  

50.                )  

51.            }  

52.         

53.            def /(that: Rational): Rational ={  

54.                new Rational(  

55.                    numer * that.denom,  

56.                    that.numer * denom  

57.                )  

58.            }  

59.              

60.            // 求倒数的前缀操作符,必须加上unary_修饰 

61.            def unary_~(that: Rational): Rational ={  

62.                new Rational(  

63.                    denom, numer  

64.                )  

65.            }  

66.         

67.            // 递归求分子分母的最大公约数,必须指明结果类型  

68.            private def gcd(a: Int, b: Int): Int =   

69.                if(b == 0) a else gcd(b, a % b)  

70.         

71.            // 有理数的比较:小于  

72.            def lessThan(that: Rational) =  

73.                numer * that.denom < that.numer * denom  

74.         

75.            // 求两数中的较大数  

76.            def max(that: Rational) : Rational =  

77.                //if(lessThan(that)) that else this //省略this也可以  

78.                ifthis lessThan that ) that else this 

79.        

 

 

说明一下代码中的几点,也便于我自己理清Scala 中的编程规范:

1  require(d != 0) Scala 中的方法、主构造器要求调用发必须满足的先决条件,是一种限制,确保了程序的正常运行。看起来功能类似于断言机制,但是 Scala 中也有断言啊,所以我目前还不明白这两者的区别所在。

2  私有的 gcd() 函数是递归函数,必须指明其结果类型(返回类型),它用于求得分子、分母的最大公约数,用于对分数的化简。此外,在 Scala 中,尾递归函数可以被 Scala 编译器进行优化,尽量写尾递归的函数,这才是 Scala 函数式编程偏向的以递归替代while循环的基础。

3  Scala 中的构造器重载使用了 this 关键字,其实是从构造器委托了主构造器的实现

4  add()方法和加号 + 都实现了 Rational 对象(有理数)的加法运算,看起来像是 C++ 中的运算符重载。其实,在 Scala 中并没有操作符的概念,就算是 1 + 2 中的加号,也是一个由Int类型的 1 所调用的函数而已。(说起来,Scala 更像是完全的面向对象语言,至少比 Java 中还存在原始类型要好)当然,我们还可以完整地定义出其他减法sub 函数等等。

5  对于求倒数操作符 ~ ,因为在这里我们将它作为一个前缀操作符,所以必须用 unary_ 修饰,告诉编译器将 ~ 看作前缀操作符。

 

有了上面一系列的函数定义(上面的代码中多是操作符定义),我们就能够进行编写一些脚本代码来测试一下 Rational 类的这些运算了,如下:

 

1.            val a = new Rational(1, 3)  

2.            val b = new Rational(1, 2)  

3.             

4.            println(a add b)  

5.            println(a.add(b))  

6.             

7.            println(a.lessThan(b))  

8.            println(a lessThan b)  

9.             

10.        println(a.max(b))  

11.        println(a max b)  

12.         

13.        val c = new Rational(3)  

14.        println(c)  

15.         

16.        println(new Rational(66, 42))  

17.         

18.        print("1/3 + 1/2 = ")  

19.        println(a + b)  

20.         

21.        print("1/3 - 1/2 = ")  

22.        println(a - b)  

23.         

24.        print("1/3 * 1/2 = ")  

25.        println(a * b)  

26.         

27.        print("(1/3) / (1/2) = ")  

28.        println(a / b) 

 

由于我把上面 Rational 类的定义和这一些针对 Rational 类测试的代码放在同一个 .scala 脚本文件中,并且 Rational 类的代码中有许多中文的注释,所以在使用 scala 命令执行这脚本时需要用选项 –encoding 指定字符集 UTF-8 ,运行结果如下:

本文转自 xxxx66yyyy 51CTO博客,原文链接:http://blog.51cto.com/haolloyin/384387,如需转载请自行联系原作者

你可能感兴趣的文章
Visual Studio 原生开发的10个调试技巧(二)
查看>>
Windows内核再次出现0Day漏洞 影响win2000到win10所有版本 反病毒软件恐成瞎子
查看>>
H3C品牌刀片系统强势首发
查看>>
【CSS系列】图像映射
查看>>
First blood
查看>>
java 冒泡排序和快速排序 实现
查看>>
SQL存储过程中的几个常见设定SET QUOTED_IDENTIFIER/NOCOUNT/XACT_ABORT ON/OFF
查看>>
Silverlight与Flash区别之一
查看>>
删除恢复Hadoop集群中的DataNode
查看>>
Silverlight 2动态创建矩形对象(附完整源代码)
查看>>
从京东技术演进看互联网企业的成长历程
查看>>
MFC ado+mysql+odbc技术分享
查看>>
js中让字符串中特定字符红色显示
查看>>
HttpClient4.5教程-第二章-连接管理
查看>>
redhat Nginx 安装
查看>>
oracle 配置监听
查看>>
上海访微软 详解Azure和S+S
查看>>
跨国巨头猛攻语音识别技术 让电脑听懂人们说话
查看>>
moosefs即将发布新版
查看>>
FOSCommentBundle功能包:运行测试
查看>>