This page is no longer maintained — Please continue to the home page at www.scala-lang.org

Unexpected Typing

5 replies
mehdi
Joined: 2008-11-09,
User offline. Last seen 3 years 50 weeks ago.

Hi,

I have created a poor man's type constructor as:

class TypeConst[T] {
type ListSub = List[X] forSome {type X >: T}
}

When I use it:

class A;
class B extends A;

val listSub1: TypeConst[B]#ListSub = List(new A);

it creates value listSub1 of type "List[Any]" but it is supposed to be
of type "List[_ >: B]"

Regards
Mehdi

extempore
Joined: 2008-12-17,
User offline. Last seen 35 weeks 3 days ago.
Re: Unexpected Typing

On Sat, Sep 17, 2011 at 11:52 AM, Mehdi wrote:
> it creates value listSub1 of type "List[Any]" but it is supposed to be
> of type "List[_ >: B]"

List[Any] and List[_ >: B] are the same type. (To see this, try to
construct an example of a List[Any] which is not a List[_ >: B], and
try the reverse as well.)

You will get the type you expect if you use an invariant container,
e.g. use Set instead of List.

mehdi
Joined: 2008-11-09,
User offline. Last seen 3 years 50 weeks ago.
Re: Unexpected Typing

I know they are same and you can assign variables of either type to
each other, but when you create a value as:

val listSub2: List[X] forSome {type X >: T} = List(new A);

the interpreter says that its type is "List[_ >: B]" which is more
readable. Why shouldn't it do the same for TypeConst[T]#ListSub?

On Sep 18, 8:11 am, Paul Phillips wrote:
> On Sat, Sep 17, 2011 at 11:52 AM, Mehdi wrote:
> > it creates value listSub1 of type "List[Any]" but it is supposed to be
> > of type "List[_ >: B]"
>
> List[Any] and List[_ >: B] are the same type.  (To see this, try to
> construct an example of a List[Any] which is not a List[_ >: B], and
> try the reverse as well.)
>
> You will get the type you expect if you use an invariant container,
> e.g. use Set instead of List.

mehdi
Joined: 2008-11-09,
User offline. Last seen 3 years 50 weeks ago.
Re: Unexpected Typing

let me correct:

val listSub2: List[X] forSome {type X >: B} = List(new A);

On Sep 18, 11:04 am, Mehdi Bahribayli wrote:
> I know they are same and you can assign variables of either type to
> each other, but when you create a value as:
>
> val listSub2: List[X] forSome {type X >: T} = List(new A);
>
> the interpreter says that its type is "List[_ >: B]" which is more
> readable. Why shouldn't it do the same for TypeConst[T]#ListSub?
>
> On Sep 18, 8:11 am, Paul Phillips wrote:
>
>
>
>
>
>
>
> > On Sat, Sep 17, 2011 at 11:52 AM, Mehdi wrote:
> > > it creates value listSub1 of type "List[Any]" but it is supposed to be
> > > of type "List[_ >: B]"
>
> > List[Any] and List[_ >: B] are the same type.  (To see this, try to
> > construct an example of a List[Any] which is not a List[_ >: B], and
> > try the reverse as well.)
>
> > You will get the type you expect if you use an invariant container,
> > e.g. use Set instead of List.

extempore
Joined: 2008-12-17,
User offline. Last seen 35 weeks 3 days ago.
Re: Re: Unexpected Typing

On Sun, Sep 18, 2011 at 12:04 AM, Mehdi Bahribayli
wrote:
> I know they are same and you can assign variables of either type to
> each other, but when you create a value as:
>
> val listSub2: List[X] forSome {type X >: T} = List(new A);
>
> the interpreter says that its type is "List[_ >: B]" which is more
> readable. Why shouldn't it do the same for TypeConst[T]#ListSub?

The compiler applies the specified simplification rules for
existentials and given a chance that type will simplify to List[Any].
It's an artifact of the repl implementation that the wildcard form
even survives long enough for you to see it.

scala> class A ; class B extends A
defined class A
defined class B

scala> val listSub2: List[X] forSome {type X >: B} = List(new A);
listSub2: List[_ >: B] = List(A@370c488c)

scala> val x = listSub2
x: List[Any] = List(A@370c488c)

scala> listSub2
res1: List[Any] = List(A@370c488c)

mehdi
Joined: 2008-11-09,
User offline. Last seen 3 years 50 weeks ago.
Re: Unexpected Typing

Thanks Paul

On Sep 19, 6:31 am, Paul Phillips wrote:
> On Sun, Sep 18, 2011 at 12:04 AM, Mehdi Bahribayli
>
> wrote:
> > I know they are same and you can assign variables of either type to
> > each other, but when you create a value as:
>
> > val listSub2: List[X] forSome {type X >: T} = List(new A);
>
> > the interpreter says that its type is "List[_ >: B]" which is more
> > readable. Why shouldn't it do the same for TypeConst[T]#ListSub?
>
> The compiler applies the specified simplification rules for
> existentials and given a chance that type will simplify to List[Any].
> It's an artifact of the repl implementation that the wildcard form
> even survives long enough for you to see it.
>
> scala> class A ; class B extends A
> defined class A
> defined class B
>
> scala> val listSub2: List[X] forSome {type X >: B} = List(new A);
> listSub2: List[_ >: B] = List(A@370c488c)
>
> scala> val x = listSub2
> x: List[Any] = List(A@370c488c)
>
> scala> listSub2
> res1: List[Any] = List(A@370c488c)

Copyright © 2012 École Polytechnique Fédérale de Lausanne (EPFL), Lausanne, Switzerland