- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
Recursive composition of a contravariant type
Mon, 2008-12-22, 19:30
Hello,
I would like to connect an output channel to another output channel to implement a form of indirection, for example to pass on messages from an Output[String] to an Output[Any]. I modeled my problem using the code snipped below. Since output channels are contravariant the compiler doesn't like my `connect` method.
trait MyOutput[-E] {
private[this] var o:MyOutput[_ >: E] = _
def connect(oc:MyOutput[_ >: E]) { // See error below
o = oc
}
def !(e:E) {
o!e
}
}
The error I get is:
<console>:8: error: contravariant type E occurs in covariant position in type MyOutput[_ >: E] of value oc
def connect(oc:MyOutput[_ >: E]) {
^
This looks simple but it's a real puzzler for me... :(
Thanks,
Sebastien
I would like to connect an output channel to another output channel to implement a form of indirection, for example to pass on messages from an Output[String] to an Output[Any]. I modeled my problem using the code snipped below. Since output channels are contravariant the compiler doesn't like my `connect` method.
trait MyOutput[-E] {
private[this] var o:MyOutput[_ >: E] = _
def connect(oc:MyOutput[_ >: E]) { // See error below
o = oc
}
def !(e:E) {
o!e
}
}
The error I get is:
<console>:8: error: contravariant type E occurs in covariant position in type MyOutput[_ >: E] of value oc
def connect(oc:MyOutput[_ >: E]) {
^
This looks simple but it's a real puzzler for me... :(
Thanks,
Sebastien
Tue, 2008-12-23, 01:37
#2
Re: Re: Recursive composition of a contravariant type
Ok, I understand why the compiler was not happy now...
Thanks a lot!
Sebastien
2008/12/23 Eric Willigers <ewilligers@gmail.com>
Thanks a lot!
Sebastien
2008/12/23 Eric Willigers <ewilligers@gmail.com>
Sebastien Bocq wrote:
Hello,
I would like to connect an output channel to another output channel to implement a form of indirection, for example to pass on messages from an Output[String] to an Output[Any]. I modeled my problem using the code snipped below. Since output channels are contravariant the compiler doesn't like my `connect` method.
trait MyOutput[-E] {
private[this] var o:MyOutput[_ >: E] = _
def connect(oc:MyOutput[_ >: E]) { // See error below
o = oc
}
def !(e:E) {
o!e
}
}
The error I get is:
<console>:8: error: contravariant type E occurs in covariant position in type MyOutput[_ >: E] of value oc
def connect(oc:MyOutput[_ >: E]) {
^
This looks simple but it's a real puzzler for me... :(
This can't be done type safely when you have mutability and aliasing.
trait MyOutput[-E] {
private[this] var o: MyOutput[_ >: E] = _
def connect(oc: Any) {
o = oc.asInstanceOf[MyOutput[_ >: E]]
}
def !(e:E) {
o!e
}
}
object Test extends Application {
val o2 = new MyOutput[String] { override def !(e:String) {} }
val o1 = new MyOutput[Any] {}
(o1 : MyOutput[String]).connect(o2)
o1 ! this // ClassCastException when we pass anything other than a String
}
Sebastien Bocq wrote:
> Hello,
>
> I would like to connect an output channel to another output channel to
> implement a form of indirection, for example to pass on messages from an
> Output[String] to an Output[Any]. I modeled my problem using the code
> snipped below. Since output channels are contravariant the compiler
> doesn't like my `connect` method.
>
> trait MyOutput[-E] {
>
> private[this] var o:MyOutput[_ >: E] = _
>
> def connect(oc:MyOutput[_ >: E]) { // See error below
> o = oc
> }
>
> def !(e:E) {
> o!e
> }
> }
>
> The error I get is:
>
> :8: error: contravariant type E occurs in covariant position in
> type MyOutput[_ >: E] of value oc
> def connect(oc:MyOutput[_ >: E]) {
> ^
>
> This looks simple but it's a real puzzler for me... :(
This can't be done type safely when you have mutability and aliasing.
trait MyOutput[-E] {
private[this] var o: MyOutput[_ >: E] = _
def connect(oc: Any) {
o = oc.asInstanceOf[MyOutput[_ >: E]]
}
def !(e:E) {
o!e
}
}
object Test extends Application {
val o2 = new MyOutput[String] { override def !(e:String) {} }
val o1 = new MyOutput[Any] {}
(o1 : MyOutput[String]).connect(o2)
o1 ! this // ClassCastException when we pass anything other than a
String
}