Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Scala: Class parameters access vs object fields access

I am coming from Java background and new to Scala, currently going through the book 'Programming in Scala'.

There is an example in the book as below:

class Rational(n: Int, d: Int) { // This won't compile
  require(d != 0)
  override def toString = n + "/" + d

  def add(that: Rational): Rational = new Rational(n * that.d + that.n * d, d * that.d)
}

However, given this code the compiler complains:

error: value d is not a member of Rational
       new Rational(n * that.d + that.n * d, d * that.d)
                             ^
error: value n is not a member of Rational
       new Rational(n * that.d + that.n * d, d * that.d)
                                      ^
error: value d is not a member of Rational
       new Rational(n * that.d + that.n * d, d * that.d)
                                                      ^

The explanation says:

Although class parameters n and d are in scope in the code of your add method, you can only access their value on the object on which add was invoked. Thus, when you say n or d in add's implementation, the compiler is happy to provide you with the values for these class parameters. But it won't let you say that.n or that.d because that does not refer to the Rational object on which add was invoked. To access the numerator and denominator on that, you'll need to make them into fields.

Also the correct implementation is given as below:

class Rational(n: Int, d: Int) {
  require(d != 0)
  val numer: Int = n
  val denom: Int = d

  override def toString = numer + "/" + denom

  def add(that: Rational): Rational =
    new Rational(
      numer * that.denom + that.numer * denom,
      denom * that.denom
    )
}

I tried to understand this many times, but still not clear.

I already have the n and d parameters at class level. I am able to access them in add method. I am passing another Rational object to the add method. It should also be having n and d, right?

What is wrong with that.n and that.d? Why do I need to take the parameters in fields?

Also, the overridden toString method is simply taking n and d, how does that not fail?

I may sound stupid, but really need to understand this clearly for better fundamentals before I move ahead.

like image 755
vijayinani Avatar asked Mar 09 '17 14:03

vijayinani


2 Answers

Parameters passed to a class constructor defaults to private members and thus are are available to all the class code (as seen in the toString override) but they are not accessible as instance members (so that.d doesn't work).

You can tell the compiler not to use the default.

class Rational(val n: Int, val d: Int) { // now it compiles
  ...

Alternatively, arguments passed to a case class default to instance members.

case class Rational(n: Int, d: Int) { // this also compiles
  ...
like image 81
jwvh Avatar answered Sep 20 '22 15:09

jwvh


Scala has more types of access modifiers than Java. In Scala there's a thing called private[this] which means "private to the current object" which is stricter than the normal private which means "private to all objects of this class".

class Rational(n: Int, d: Int)

is basically the same as

class Rational(private[this] val n: Int, private[this] val d: Int)

At a higher, almost philosophical level you could say that

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

is like a static method returning a Rational and it has parameters n and d, and as in any method its parameters are local to the method. By qualifying the parameters with val or var you turn those parameters into fields of the Rational, without having to write them twice: once as a parameter to the method (or constructor, which is the more specific name for this static method), and once as a field.

like image 32
Jasper-M Avatar answered Sep 21 '22 15:09

Jasper-M