2011-07-28 28 views
2

我写它的声明看起来像一个2D Vector类:蕴含太多!

case class Vec2(x:Float, y:Float) extends (Float, Float)(x, y) { 

    def +(v:Vec2) = Vec2(v.x+x, v.y+y) 
    //Subtract, dot product, projection, etc. 
    ... 
    ... 
} 

我希望能够写东西像Vec2(3, 7) + (2, 9)所以我写

scala> implicit def ii2v2(i:(Int, Int)) = Vec2(i._1, i._2) 
ii2v2: (i: (Int, Int))org.zhang.lib.misc.Vec2 

scala> Vec2(2, 6) + (3, 1) 
res25: org.zhang.lib.misc.Vec2 = (5.0,7.0) 

大。但是,如果我尝试Vec2(3, 7) + (2.6f, 9.3f),隐含不起作用,因为(Float, Float)(Int, Int)不匹配。我想出的唯一解决方案是写出4个implicits,为(Int,Int), (Int, Float), (Float, Int), and (Float, Float)

当您尝试考虑双打时,或者当您编写Vec3类时,问题变得荒谬。有没有解决的办法?我可以只是Vec2-ify一切,但我的一部分只是真的想添加一个(Int,Int)到一个Vec2 :)

回答

11

这样做:

implicit def ii2v2[T: Numeric, U: Numeric](i:(T, U)) = { 
    import Numeric.Implicits._ 
    Vec2(i._1.toFloat, i._2.toFloat) 
} 

此背景下使用边界来告诉编译器寻找一个数字[T]的范围,其存在的数值类型。自2.9起可用的进口Numeric.Implicits._允许编写toFloat

在2.8,你可以写:

implicit def ii2v2[T, U](i:(T, U))(implicit num1: Numeric[T], num2: Numeric[U]) = { 
    Vec2(num1.toFloat(i._1), num2.toFloat(i._2)) 
} 

看到这个的其他问题是相似的:Writing a generic mean function in Scala

+0

酷的东西!我将不得不更多地关注数字。谢谢! –