- About Scala
- Documentation
- Code Examples
- Software
- Scala Developers
matching case classes with type parameters?
Sat, 2009-01-31, 18:07
I have a newbie question that perhaps someone can help me with. I'm
trying to have a base class with a type parameter and a pattern match:
abstract class X[T](var a : T)
{
// some type independent code (omitted)...
// type dependent
def field2val(field : String)
{
this match
{
case floatX(_) => a = field.toFloat;
case intX(_) => a = field.toInt;
}
}
}
case class floatX(a : Float) extends X[Float](a);
case class intX(a : Int) extends X[Int](a);
this leads to compile time errors with constructor errors and type
errors for each case:
test.scala:7: error: constructor cannot be instantiated to expected type;
found : floatX
required: X[T]
case floatX(_) => a = field.toFloat;
^
test.scala:7: error: type mismatch;
found : Float
required: T
case floatX(_) => a = field.toFloat;
...
Is there a more "correct" way to handle this? Basically, I want type
specialization for some small set of methods, but most are type
independent/generic.
thanks
Sat, 2009-01-31, 19:17
#2
Re: matching case classes with type parameters?
What you are trying to do is pretty crazy, but not impossible.
abstract class X[T](var a : T)
{
def field2val(field : String) : Any = {
val self : AnyRef = this
self match {
case selff: floatX => selff.a = field.toFloat
case selfi: intX => selfi.a = field.toInt
}
}
}
case class floatX(param : Float) extends X[Float](param)
case class intX(param : Int) extends X[Int](param)
The question is: why not use abstract methods and overriding? You haven't sealed the base class X, so I can write arbitrary subclasses that wont' be caught by the match.
On Sat, Jan 31, 2009 at 9:07 AM, Swade Leather <swadenator@gmail.com> wrote:
abstract class X[T](var a : T)
{
def field2val(field : String) : Any = {
val self : AnyRef = this
self match {
case selff: floatX => selff.a = field.toFloat
case selfi: intX => selfi.a = field.toInt
}
}
}
case class floatX(param : Float) extends X[Float](param)
case class intX(param : Int) extends X[Int](param)
The question is: why not use abstract methods and overriding? You haven't sealed the base class X, so I can write arbitrary subclasses that wont' be caught by the match.
On Sat, Jan 31, 2009 at 9:07 AM, Swade Leather <swadenator@gmail.com> wrote:
I have a newbie question that perhaps someone can help me with. I'm
trying to have a base class with a type parameter and a pattern match:
abstract class X[T](var a : T)
{
// some type independent code (omitted)...
// type dependent
def field2val(field : String)
{
this match
{
case floatX(_) => a = field.toFloat;
case intX(_) => a = field.toInt;
}
}
}
case class floatX(a : Float) extends X[Float](a);
case class intX(a : Int) extends X[Int](a);
this leads to compile time errors with constructor errors and type
errors for each case:
test.scala:7: error: constructor cannot be instantiated to expected type;
found : floatX
required: X[T]
case floatX(_) => a = field.toFloat;
^
test.scala:7: error: type mismatch;
found : Float
required: T
case floatX(_) => a = field.toFloat;
...
Is there a more "correct" way to handle this? Basically, I want type
specialization for some small set of methods, but most are type
independent/generic.
thanks
Use regular OOP. Leave field2val abstract and implement it in each
concrete subclass.
Using matching for this is like swatting flies with a sledgehammer :-)
--Jens [via iPhone]
On Jan 31, 2009, at 9:07 AM, Swade Leather wrote:
> I have a newbie question that perhaps someone can help me with. I'm
> trying to have a base class with a type parameter and a pattern match:
>
> abstract class X[T](var a : T)
> {
> // some type independent code (omitted)...
>
> // type dependent
> def field2val(field : String)
> {
> this match
> {
> case floatX(_) => a = field.toFloat;
> case intX(_) => a = field.toInt;
> }
> }
> }
>
> case class floatX(a : Float) extends X[Float](a);
> case class intX(a : Int) extends X[Int](a);
>
> this leads to compile time errors with constructor errors and type
> errors for each case:
>
> test.scala:7: error: constructor cannot be instantiated to expected
> type;
> found : floatX
> required: X[T]
> case floatX(_) => a = field.toFloat;
> ^
> test.scala:7: error: type mismatch;
> found : Float
> required: T
> case floatX(_) => a = field.toFloat;
> ...
>
> Is there a more "correct" way to handle this? Basically, I want type
> specialization for some small set of methods, but most are type
> independent/generic.
>
> thanks