Scala的私有字段和定义操作符
创始人
2024-04-18 15:40:28
0

私有字段和方法

上一个版本的Rational类里,我们只是分别用n初始化了numer,用d初始化了denom。结果,Rational的分子和分母可能比它所需要的要大。例如分数66/42 ,可以更约简化为相同的最简形式,11/7 ,但Rational的主构造器当前并不做这个工作:

51CTO编辑推荐:Scala编程语言专题

  1. scala> new Rational(66, 42)  
  2. res15: Rational = 66/42 
要想对分数进行约简化,需要把分子和分母都除以***公约数:greatest common divisor。如:66和42的***公约数是6。(另一种说法就是,6是能够除尽66和42的***的整数。)66/42 的分子和分母都除以6就产生它的最简形式,11/7 。代码6.3展示了如何做到这点:

  1. class Rational(n: Int, d: Int) {  
  2.  require(d != 0)  
  3.  private val g = gcd(n.abs, d.abs)  
  4.  val numer = n / g  
  5.  val denom = d / g  
  6.  def this(n: Int) = this(n, 1)  
  7.  def add(that: Rational): Rational =  
  8.   new Rational(  
  9.    numer * that.denom + that.numer * denom,  
  10.    denom * that.denom  
  11.   )  
  12.  override def toString = numer+"/"+denom  
  13.  private def gcd(a: Int, b: Int): Int =  
  14.   if (b == 0) a else gcd(b, a % b)  
  15. }  
代码 6.3 带私有字段和方法的Rational

这个版本的Rational里,我们添加了私有字段,g,并修改了numer和denom的初始化器(初始化器:initializer是初始化变量,例如初始化numer的“n / g”,的代码)。因为g是私有的,它只能在类的主体之内,而不能在外部被访问。我们还添加了一个私有方法,gcd,用来计算传入的两个Int的***公约数。比方说,gcd(12, 8)是4。正如你在4.1节中看到的,想让一个字段或方法私有化你只要把private关键字放在定义的前面。私有的“助手方法”gcd的目的是把类的其它部分,这里是主构造器,需要的代码分离出来。为了确保g始终是正的,我们传入n和d的绝对值,调用abs即可获得任意整数的绝对值。

Scala编译器将把Rational的三个字段的初始化代码依照它们在源代码中出现的次序放入主构造器。所以g的初始化代码,gcd(n.abs, d.abs),将在另外两个之前执行,因为它在源文件中出现得最早。g将被初始化为类参数,n和d,的绝对值的***公约数。然后再被用于numer和denom的初始化。通过把n和d整除它们的***公约数,g,每个Rational都将被构造成它的最简形式:

  1. scala> new Rational(66, 42)  
  2. res24: Rational = 11/7 
定义操作符

Rational加法的当前实现仅就完成功能来讲是没问题的,但它可以做得更好用。你或许会问你自己为什么对于整数或浮点数你可以写成:

  1. x + y 
但是如果是分数就必须写成:

  1. x.add(y) 
或至少是:

  1. x add y 
没有合理的解释为什么就必须是这样的。分数和别的数应该是一样的。数学的角度上看他们甚至比,唔,浮点数,更自然。为什么就不能使用自然的数学操作符呢?Scala里面你做得到。本章后续部分,我们会告诉你怎么做。

***步是用通常的数学的符号替换add方法。这可以直接做到,因为Scala里+是合法的标识符。我们可以用+定义方法名。既然已经到这儿了,你可以同样实现一个*方法以实现乘法,结果展示在代码6.4中:

  1. class Rational(n: Int, d: Int) {  
  2.  require(d != 0)  
  3.  private val g = gcd(n.abs, d.abs)  
  4.  val numer = n / g  
  5.  val denom = d / g  
  6.  def this(n: Int) = this(n, 1)  
  7.  def +(that: Rational): Rational =  
  8.   new Rational(  
  9.    numer * that.denom + that.numer * denom,  
  10.    denom * that.denom  
  11.   )  
  12.  def *(that: Rational): Rational =  
  13.   new Rational(numer * that.numer, denom * that.denom)  
  14.  override def toString = numer+"/"+denom  
  15.  private def gcd(a: Int, b: Int): Int =  
  16.   if (b == 0) a else gcd(b, a % b)  
  17. }  
代码 6.4 带操作符方法的Rational

有了这种方式定义的Rational类,你现在可以这么写了:

  1. scala> val x = new Rational(1, 2)  
  2. x: Rational = 1/2 
  3. scala> val y = new Rational(2, 3)  
  4. y: Rational = 2/3 
  5. scala> x + y  
  6. res32: Rational = 7/6 
与以往一样,在***输入的那行里的语法格式相等于一个方法调用。你也能这么写:

  1. scala> x.+(y)  
  2. res33: Rational = 7/6 
不过这样写可读性不佳。

另外一件要提的是基于5.8节中提到的Scala的操作符优先级规则,Rational里面的*方法要比+方法绑定得更结实。或者说,Rational涉及到+和*操作的表达式会按照预期的方式那样表现。例如,x + x * y会当作x + (x * y)而不是(x + x) * y:

  1. scala> x + x * y  
  2. res34: Rational = 5/6 
  3. scala> (x + x) * y  
  4. res35: Rational = 2/3 
  5. scala> x + (x * y)  
  6. res36: Rational = 5/6 

【相关阅读】

  1. Scala的从构造器:主构造器之外的构造器
  2. 在Scala中检查先决条件、添加字段和自指向
  3. Scala Rational对象的toString方法
  4. 学习Scala中的Rational类:分数的模型化
  5. Scala中的富包装器:富操作和富类列表

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...