- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
2.8.0.head Structured Type and isInstanceOf
Fri, 2010-04-02, 16:32
Is there a problem with isInstanceOf and Structured Types?
shell> scala -unchecked
Welcome to Scala version 2.8.0.Beta1-prerelease (Java HotSpot(TM) 64-Bit
Server
VM, Java 1.6.0_19).
Type in expressions to have them evaluated.
Type :help for more information.
scala> class F {
| def getName(): String = {
| "hi"
| }
| }
defined class F
scala> var f = new F()
f: F = F@6b91602
scala> f.isInstanceOf[{ def getName(): String }]
:7: warning: refinement AnyRef{def getName: <?>} in type AnyRef{def
getName: <?>} is unchecked since it is eliminated by erasure
f.isInstanceOf[{ def getName(): String }]
^
res0: Boolean = true
// Note: returns true
scala> "someString".isInstanceOf[{ def getName(): String }]
:5: warning: refinement AnyRef{def getName: <?>} in type AnyRef{def
getName: <?>} is unchecked since it is eliminated by erasure
"someString".isInstanceOf[{ def getName(): String }]
^
res1: Boolean = true
// Note: should not return true
scala> var a = new Array[{ def getName(): String}](1)
a: Array[AnyRef{def getName(): String}] = Array(null)
scala> a(0) = f
scala> a(0)
res3: AnyRef{def getName(): String} = F@6b91602
scala> a(0) = "someString"
:6: error: type mismatch;
found : java.lang.String("someString")
required: AnyRef{def getName(): String}
a(0) = "someString"
^
// So, adding to the Array fails as it should,
// but isInstanceOf does not work
Richard
Fri, 2010-04-02, 17:07
#2
Re: 2.8.0.head Structured Type and isInstanceOf
Le 02/04/2010 17:32, richard emberson a écrit :
[...]
> scala> "someString".isInstanceOf[{ def getName(): String }]
> :5: warning: refinement AnyRef{def getName: <?>} in type
> AnyRef{def
> getName: <?>} is unchecked since it is eliminated by erasure
> "someString".isInstanceOf[{ def getName(): String }]
> ^
The warning is really important and say exactly what is happening. You
are just testing that you string is an instance of AnyRef.
Fri, 2010-04-02, 17:27
#3
Re: 2.8.0.head Structured Type and isInstanceOf
On Friday April 2 2010, Francois wrote:
> Le 02/04/2010 17:32, richard emberson a écrit :
> [...]
>
>> scala> f.isInstanceOf[{ def getName(): String }]
>> :7: warning: refinement AnyRef{def getName: <?>} in type AnyRef{def getName: <?>} is unchecked since it is eliminated by erasure
>> f.isInstanceOf[{ def getName(): String }]
>
> The warning is really important and say exactly what is happening.
> You are just testing that you string is an instance of AnyRef.
One could make the argument that for what is ordinarily a dynamic test,
the generated code should test the instance for the presence of a
structurally compatible method, analogous to the way an attempt to
invoke that method would use reflection to find the method.
Randall Schulz
Fri, 2010-04-02, 17:37
#4
Re: 2.8.0.head Structured Type and isInstanceOf
import java.io.File
import collection.mutable.ListBuffer
object ST {
class T(name: String) {
def getName(): String = name
override def toString(): String = {
"T.name=" +name
}
}
def main(args: Array[String]) {
val l = List[AnyRef]("hi", new File("test.txt"), new T("someT"))
val lb = new ListBuffer[{ def getName(): String }]
for (v <- l) {
if (v.isInstanceOf[ {def getName(): String } ]) {
lb += v.asInstanceOf[{ def getName(): String }]
}
}
println(lb)
}
}
shell> scalac ST.scala
warning: there were unchecked warnings; re-run with -unchecked for details
one warning found
shell> scala ST
ListBuffer(hi, test.txt, T.name=someT)
This is not the expected result which is to say,
its a run-time error.
Great there was a compile-time warning.
The result was a run-time error.
The best solution would be for isInstanceOf to work
at run-time. Next best is for asInstanceOf to work
at run-time. And I mean "work" as a developer might
expect rather than "work" only to a point due to
limitations in the platform. The third best option
is for the compiler to create a compile-time error
stating that run-time information concerning
Structured Types is very limited and isInstanceOf
and asInstanceOf can not be used with them.
Richard
On 04/02/2010 09:11 AM, Randall R Schulz wrote:
> On Friday April 2 2010, Francois wrote:
>> Le 02/04/2010 17:32, richard emberson a écrit :
>> [...]
>>
>>> scala> f.isInstanceOf[{ def getName(): String }]
>>> :7: warning: refinement AnyRef{def getName:<?>} in type AnyRef{def getName:<?>} is unchecked since it is eliminated by erasure
>>> f.isInstanceOf[{ def getName(): String }]
>>
>> The warning is really important and say exactly what is happening.
>> You are just testing that you string is an instance of AnyRef.
>
> One could make the argument that for what is ordinarily a dynamic test,
> the generated code should test the instance for the presence of a
> structurally compatible method, analogous to the way an attempt to
> invoke that method would use reflection to find the method.
>
>
> Randall Schulz
>
Fri, 2010-04-02, 19:27
#5
Re: 2.8.0.head Structured Type and isInstanceOf
I'm sympathetic but sometimes casting across type erasure is useful so a warning is more appropriate than an error.
That said, there is enough information in the class file for isInstanceOf and asInstanceOf to do -SOME- runtime checking of structural types, certainly more than is being done now. Since this is very important to you would you mind specifying the details of the enhancement (exactly what will and will not be checked, how it all behaves with subtyping, how it all behaves with Scala's various boxed forms, etc) in a SID?
On Fri, Apr 2, 2010 at 9:23 AM, richard emberson <richard.emberson@gmail.com> wrote:
That said, there is enough information in the class file for isInstanceOf and asInstanceOf to do -SOME- runtime checking of structural types, certainly more than is being done now. Since this is very important to you would you mind specifying the details of the enhancement (exactly what will and will not be checked, how it all behaves with subtyping, how it all behaves with Scala's various boxed forms, etc) in a SID?
On Fri, Apr 2, 2010 at 9:23 AM, richard emberson <richard.emberson@gmail.com> wrote:
Great there was a compile-time warning.
The result was a run-time error.
The best solution would be for isInstanceOf to work
at run-time. Next best is for asInstanceOf to work
at run-time. And I mean "work" as a developer might
expect rather than "work" only to a point due to
limitations in the platform. The third best option
is for the compiler to create a compile-time error
stating that run-time information concerning
Structured Types is very limited and isInstanceOf
and asInstanceOf can not be used with them.
Richard
Fri, 2010-04-02, 21:37
#6
Re: 2.8.0.head Structured Type and isInstanceOf
On Fri, Apr 02, 2010 at 11:25:58AM -0700, James Iry wrote:
> That said, there is enough information in the class file for
> isInstanceOf and asInstanceOf to do -SOME- runtime checking of
> structural types, certainly more than is being done now.
I've had this 75% done for a while. The static side is done, but I have
to generalize the reflection machinery in cleanup to do the runtime
check cleanly. IOW in that branch this gives correct answers more often
than a stopped clock:
"abc".isInstanceOf[{ def length(): Int }]
But this still needs work:
("abc": AnyRef).isInstanceOf[{ def length(): Int }]
On Fri, Apr 02, 2010 at 08:32:36AM -0700, richard emberson wrote:
> Is there a problem with isInstanceOf and Structured Types?
If only there were some sort of warning!