- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
A seemingly strange question
Mon, 2010-04-26, 22:28
Bonjour Scalars,
Given a
def foo[T](bar : Manifest[T], baz : Any) = ...
What'd be the simplest way to determine whether said bar is assignable from said foo?
(Gotto work for primitives, arrays and other assorted objects)
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
Given a
def foo[T](bar : Manifest[T], baz : Any) = ...
What'd be the simplest way to determine whether said bar is assignable from said foo?
(Gotto work for primitives, arrays and other assorted objects)
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
Tue, 2010-04-27, 01:47
#2
Re: A seemingly strange question
This is what I would come up with, not really a thing of beauty, not really a simple way:
def foo[T](x: Any)(implicit mf: Manifest[T]) = mf == manifest[Any] || {
def primitive(c: Class[_]) = mf == manifest[AnyVal] || mf.erasure == c
x match {
case _:Int => primitive(classOf[Int])
case _:Unit => primitive(classOf[Unit])
case _:Long => primitive(classOf[Long])
// ... other AnyVals
case null =>
!(mf.erasure.isPrimitive || classOf[NotNull].isAssignableFrom(mf.erasure))
case x:AnyRef => mf.erasure.isAssignableFrom(x.getClass)
}
}
Works for all AnyVals, arrays and unparametrized types.
Regards,
Moritz
Am 26.04.2010 um 23:27 schrieb Viktor Klang:
> Bonjour Scalars,
>
> Given a
>
> def foo[T](bar : Manifest[T], baz : Any) = ...
>
> What'd be the simplest way to determine whether said bar is assignable from
> said foo?
>
> (Gotto work for primitives, arrays and other assorted objects)
>
Tue, 2010-04-27, 02:57
#3
Re: A seemingly strange question
On Mon, Apr 26, 2010 at 11:27:58PM +0200, Viktor Klang wrote:
> def foo[T](bar : Manifest[T], baz : Any) = ...
>
> What'd be the simplest way to determine whether said bar is assignable from
> said foo?
>
> (Gotto work for primitives, arrays and other assorted objects)
No time this moment to write you a thrilling implementation, but
depending on what you mean by "work for primitives" it is not possible
as you have framed it. You can't tell whether an "Any" parameter was a
primitive Int or a java.lang.Integer unless you take additional measures
which are not present in that signature.
Tue, 2010-04-27, 03:57
#4
Re: A seemingly strange question
On Mon, Apr 26, 2010 at 11:27:58PM +0200, Viktor Klang wrote:
> def foo[T](bar : Manifest[T], baz : Any) = ...
So again depending on what you're trying to do, you might look at the
code that I used to have in trunk in scala.reflect.Invocation, which I
pulled because I had to lower the surface area of the stuff I'm trying
to get in good shape for 2.8. But here is how I wrote a generic
interface which maintained the distinction between primitives and the
boxed versions.
Take a time machine to see the original file for how it's used in
context. But these days I would just use manifests.
https://lampsvn.epfl.ch/trac/scala/browser/scala/trunk/src/library/scala...
/**
* In order to encapsulate anything to do with reflection, we must
* overcome an issue with the boxing of primitives. If we declare a
* method which takes arguments of type Any
, by the time the
* method parameters can be examined, the primitives have already been boxed.
* The reflective call will then fail because classOf[java.lang.Integer]
* is not the same thing as classOf[scala.Int].
*
* Any useful workaround will require examining the arguments before
* the method is called. The approach here is to define two implicits,
* one for AnyRef
's and one for AnyVal
's, and
* box them in a container which preserves their original class identity.
*
Tue, 2010-04-27, 10:07
#5
Re: A seemingly strange question
On Tue, Apr 27, 2010 at 4:51 AM, Paul Phillips <paulp@improving.org> wrote:
On Mon, Apr 26, 2010 at 11:27:58PM +0200, Viktor Klang wrote:
> def foo[T](bar : Manifest[T], baz : Any) = ...
So again depending on what you're trying to do, you might look at the
code that I used to have in trunk in scala.reflect.Invocation, which I
pulled because I had to lower the surface area of the stuff I'm trying
to get in good shape for 2.8. But here is how I wrote a generic
interface which maintained the distinction between primitives and the
boxed versions.
The problem is erasure, I have a list of a parameterized type, something like: List[MyType[_]]
And for each of these, I'm going to try to set a value into it, if that value is compatible with the original tpye the MyType was parameterized with.
class MyType[T : Manifest]{
val myType = manifest[T]
var myTypeValue : Option[T] = None
def set(t : T) = ... //T Here will be unknown because of the unknown type in the list
def set(a : Any) = ... //Set value a into myTypeValue if, and only if, said "a" is assiblable from the type T as specified by the manifest "myType"
}
Does any of this make sense?
Take a time machine to see the original file for how it's used in
context. But these days I would just use manifests.
https://lampsvn.epfl.ch/trac/scala/browser/scala/trunk/src/library/scala/reflect/Invocation.scala?rev=19692
/** <p>
* In order to encapsulate anything to do with reflection, we must
* overcome an issue with the boxing of primitives. If we declare a
* method which takes arguments of type <code>Any</code>, by the time the
* method parameters can be examined, the primitives have already been boxed.
* The reflective call will then fail because <code>classOf[java.lang.Integer]</code>
* is not the same thing as <code>classOf[scala.Int].</code>
* </p>
* <p>
* Any useful workaround will require examining the arguments before
* the method is called. The approach here is to define two implicits,
* one for <code>AnyRef</code>'s and one for <code>AnyVal</code>'s, and
* box them in a container which preserves their original class identity.
* </p>
*/
trait PrimitivePreserver[T] {
val value: T
val clazz: JClass[_]
}
case class PreservedAnyVal[T <: AnyVal](value: T) extends PrimitivePreserver[T] {
val clazz = getAnyValClass(value)
}
case class PreservedAnyRef[T <: AnyRef](value: T) extends PrimitivePreserver[T] {
val clazz = value.getClass
}
implicit def makePreservedAnyRef[T <: AnyRef](x: T) = PreservedAnyRef(x)
implicit def makePreservedAnyVal[T <: AnyVal](x: T) = PreservedAnyVal(x)
--
Paul Phillips | Where there's smoke, there's mirrors!
Apatheist |
Empiricist |
up hill, pi pals! |----------* http://www.improving.org/paulp/ *----------
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
Tue, 2010-04-27, 10:47
#6
Re: A seemingly strange question
On Tue, Apr 27, 2010 at 11:01 AM, Viktor Klang <viktor.klang@gmail.com> wrote:
On Tue, Apr 27, 2010 at 4:51 AM, Paul Phillips <paulp@improving.org> wrote:On Mon, Apr 26, 2010 at 11:27:58PM +0200, Viktor Klang wrote:
> def foo[T](bar : Manifest[T], baz : Any) = ...
So again depending on what you're trying to do, you might look at the
code that I used to have in trunk in scala.reflect.Invocation, which I
pulled because I had to lower the surface area of the stuff I'm trying
to get in good shape for 2.8. But here is how I wrote a generic
interface which maintained the distinction between primitives and the
boxed versions.
The problem is erasure, I have a list of a parameterized type, something like: List[MyType[_]]
And for each of these, I'm going to try to set a value into it, if that value is compatible with the original tpye the MyType was parameterized with.
class MyType[T : Manifest]{
val myType = manifest[T]
var myTypeValue : Option[T] = None
def set(t : T) = ... //T Here will be unknown because of the unknown type in the list
def set(a : Any) = ... //Set value a into myTypeValue if, and only if, said "a" is assiblable from the type T as specified by the manifest "myType"
}
Does any of this make sense?
Also, to throw more stuff into the mix, the instances will be transported over different VMs.
Basically, what I want to know, is that given a manifest and an Any, is that Any assignable from that manifest type.
Take a time machine to see the original file for how it's used in
context. But these days I would just use manifests.
https://lampsvn.epfl.ch/trac/scala/browser/scala/trunk/src/library/scala/reflect/Invocation.scala?rev=19692
/** <p>
* In order to encapsulate anything to do with reflection, we must
* overcome an issue with the boxing of primitives. If we declare a
* method which takes arguments of type <code>Any</code>, by the time the
* method parameters can be examined, the primitives have already been boxed.
* The reflective call will then fail because <code>classOf[java.lang.Integer]</code>
* is not the same thing as <code>classOf[scala.Int].</code>
* </p>
* <p>
* Any useful workaround will require examining the arguments before
* the method is called. The approach here is to define two implicits,
* one for <code>AnyRef</code>'s and one for <code>AnyVal</code>'s, and
* box them in a container which preserves their original class identity.
* </p>
*/
trait PrimitivePreserver[T] {
val value: T
val clazz: JClass[_]
}
case class PreservedAnyVal[T <: AnyVal](value: T) extends PrimitivePreserver[T] {
val clazz = getAnyValClass(value)
}
case class PreservedAnyRef[T <: AnyRef](value: T) extends PrimitivePreserver[T] {
val clazz = value.getClass
}
implicit def makePreservedAnyRef[T <: AnyRef](x: T) = PreservedAnyRef(x)
implicit def makePreservedAnyVal[T <: AnyVal](x: T) = PreservedAnyVal(x)
--
Paul Phillips | Where there's smoke, there's mirrors!
Apatheist |
Empiricist |
up hill, pi pals! |----------* http://www.improving.org/paulp/ *----------
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
Tue, 2010-04-27, 11:27
#7
Re: A seemingly strange question
On Tue, Apr 27, 2010 at 2:45 AM, Moritz Uhlig <moritz.uhlig@jsolution.de> wrote:
This is what I would come up with, not really a thing of beauty, not really a simple way:
def foo[T](x: Any)(implicit mf: Manifest[T]) = mf == manifest[Any] || {
def primitive(c: Class[_]) = mf == manifest[AnyVal] || mf.erasure == c
x match {
case _:Int => primitive(classOf[Int])
case _:Unit => primitive(classOf[Unit])
case _:Long => primitive(classOf[Long])
// ... other AnyVals
case null =>
!(mf.erasure.isPrimitive || classOf[NotNull].isAssignableFrom(mf.erasure))
case x:AnyRef => mf.erasure.isAssignableFrom(x.getClass)
}
}
Works for all AnyVals, arrays and unparametrized types.
Thanks for the example, I'll take it for a spin and see if it'll work.
More suggestions are needed tough! :-)
Regards,
Moritz
Am 26.04.2010 um 23:27 schrieb Viktor Klang:
> Bonjour Scalars,
>
> Given a
>
> def foo[T](bar : Manifest[T], baz : Any) = ...
>
> What'd be the simplest way to determine whether said bar is assignable from
> said foo?
>
> (Gotto work for primitives, arrays and other assorted objects)
>
> --
> Viktor Klang
> | "A complex system that works is invariably
> | found to have evolved from a simple system
> | that worked." - John Gall
>
> Akka - the Actor Kernel: Akkasource.org
> Twttr: twitter.com/viktorklang
--
Viktor Klang
| "A complex system that works is invariably
| found to have evolved from a simple system
| that worked." - John Gall
Akka - the Actor Kernel: Akkasource.org
Twttr: twitter.com/viktorklang
def foo[T](bar: Manifest[T], baz: Any) = bar.erasure.isAssignableFrom(baz.getClass)
Or maybe invert bar.erasure and baz.getClass. I always found isAssignableFrom unintuitive.
On Mon, Apr 26, 2010 at 6:27 PM, Viktor Klang <viktor.klang@gmail.com> wrote:
--
Daniel C. Sobral
I travel to the future all the time.