这是一个相当不愉快的意外:
scala> Set(1, 2, 3, 4, 5)
res18: scala.collection.immutable.Set[Int] = Set(4, 5, 1, 2, 3)
scala> Set(1, 2, 3, 4, 5).toList
res25: List[Int] = List(5, 1, 2, 3, 4)
这个例子本身对我的问题给出了“否”的答案。那么呢ListSet
?
scala> import scala.collection.immutable.ListSet
scala> ListSet(1, 2, 3, 4, 5)
res21: scala.collection.immutable.ListSet[Int] = Set(1, 2, 3, 4, 5)
这似乎有效,但我应该依赖这种行为吗?
还有什么其他数据结构适合于必须保留原始顺序的唯一项的不可变集合?
顺便说一句,我确实知道distict
中的方法List
。问题是,我想在接口级别强制项目的唯一性(同时保留顺序),因此使用distinct
会弄乱我整洁的设计..
EDIT
ListSet
似乎也不是很可靠:
scala> ListSet(1, 2, 3, 4, 5).toList
res28: List[Int] = List(5, 4, 3, 2, 1)
EDIT2
在寻找完美设计的过程中,我尝试了以下方法:
scala> class MyList[A](list: List[A]) { val values = list.distinct }
scala> implicit def toMyList[A](l: List[A]) = new MyList(l)
scala> implicit def fromMyList[A](l: MyList[A]) = l.values
这实际上有效:
scala> val l1: MyList[Int] = List(1, 2, 3)
scala> l1.values
res0: List[Int] = List(1, 2, 3)
scala> val l2: List[Int] = new MyList(List(1, 2, 3))
l2: List[Int] = List(1, 2, 3)
但问题是我不想暴露MyList
图书馆外。有什么办法可以在覆盖时进行隐式转换吗?例如:
trait T { def l: MyList[_] }
object O extends T { val l: MyList[_] = List(1, 2, 3) }
scala> O.l mkString(" ") // Let's test the implicit conversion
res7: String = 1 2 3
我想这样做:
object O extends T { val l = List(1, 2, 3) } // Doesn't work