- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
A little generics/inference question
Sun, 2010-08-08, 08:28
Hi,
Can someone please help me understand why call "A1" and "A2" to generic methods foo()/bar() compile even though they don't pass the argument needed?
------------------------------------------------------
class Test {
def test() {
foo() // A1
bar() // A2
}
def foo[T](y:T):T = {
println("foo called with " + y)
y
}
def bar[T](y:T) {
println("bar called with " + y)
y
}
}
------------------------------------------------------
rgds,
Roshan
Can someone please help me understand why call "A1" and "A2" to generic methods foo()/bar() compile even though they don't pass the argument needed?
------------------------------------------------------
class Test {
def test() {
foo() // A1
bar() // A2
}
def foo[T](y:T):T = {
println("foo called with " + y)
y
}
def bar[T](y:T) {
println("bar called with " + y)
y
}
}
------------------------------------------------------
rgds,
Roshan
Sun, 2010-08-08, 09:07
#2
Re: A little generics/inference question
Oh, you mean foo() has been taken as foo(())?
A corner grammar case perhaps - where foo() was chosen to mean foo(()) (i.e., parentheses-less call "foo ()") instead of foo(<no_arg>).
Thanks,
Roshan
On Sun, Aug 8, 2010 at 1:18 PM, Stephen Tu <steve33671@gmail.com> wrote:
A corner grammar case perhaps - where foo() was chosen to mean foo(()) (i.e., parentheses-less call "foo ()") instead of foo(<no_arg>).
Thanks,
Roshan
On Sun, Aug 8, 2010 at 1:18 PM, Stephen Tu <steve33671@gmail.com> wrote:
$ scalac -Xprint:typer test1.scala
[[syntax trees at end of typer]]// Scala source: test1.scala
package <empty> {
class Test extends java.lang.Object with ScalaObject {
def this(): Test = {
Test.super.this();
()
};
def test(): Unit = {
Test.this.foo[Unit](());
Test.this.bar[Unit](())
};
def foo[T >: Nothing <: Any](y: T): T = {
scala.this.Predef.println("foo called with ".+(y));
y
};
def bar[T >: Nothing <: Any](y: T): Unit = {
scala.this.Predef.println("bar called with ".+(y));
{
y;
()
}
}
}
}
so type Unit was inferred, and since () is unit, it compiles. here's the output if you call test():
scala> val t = new Test
t: Test = Test@1c9fe7e
scala> t.test
foo called with ()
bar called with ()
Sun, 2010-08-08, 10:07
#3
Re: A little generics/inference question
On Sun, Aug 8, 2010 at 1:06 AM, Roshan Dawrani <roshandawrani@gmail.com> wrote:
I could be wrong, but I think its more to do with, in your specific case, interpreting the () as Unit instead of a function application with no arguments is the only way to make the expression type check correctly. it definitely isn't an assumption tied to the grammar, since then you could never call a function of no arugments.
Oh, you mean foo() has been taken as foo(())?
A corner grammar case perhaps - where foo() was chosen to mean foo(()) (i.e., parentheses-less call "foo ()") instead of foo(<no_arg>).
I could be wrong, but I think its more to do with, in your specific case, interpreting the () as Unit instead of a function application with no arguments is the only way to make the expression type check correctly. it definitely isn't an assumption tied to the grammar, since then you could never call a function of no arugments.
[[syntax trees at end of typer]]// Scala source: test1.scala
package <empty> {
class Test extends java.lang.Object with ScalaObject {
def this(): Test = {
Test.super.this();
()
};
def test(): Unit = {
Test.this.foo[Unit](());
Test.this.bar[Unit](())
};
def foo[T >: Nothing <: Any](y: T): T = {
scala.this.Predef.println("foo called with ".+(y));
y
};
def bar[T >: Nothing <: Any](y: T): Unit = {
scala.this.Predef.println("bar called with ".+(y));
{
y;
()
}
}
}
}
so type Unit was inferred, and since () is unit, it compiles. here's the output if you call test():
scala> val t = new Test
t: Test = Test@1c9fe7e
scala> t.test
foo called with ()
bar called with ()