List
has 2 methods that are specified to prepend an element to an (immutable) list:
+:
(implementingSeq.+:
), and::
(defined only inList
)
+:
technically has a more general type signature—
def +:[B >: A, That](elem: B)(implicit bf: CanBuildFrom[List[A], B, That]): That
def ::[B >: A](x: B): List[B]
—but ignoring the implicit, which according to the doc message merely requires That
to be List[B]
, the signatures are equivalent.
What is the difference between List.+:
and List.::
? If they are in fact identical, I assume +:
would be preferred to avoid depending on the concrete implementation List
. But why was another public method defined, and when would client code call it?
Edit
There is also an extractor for ::
in pattern matching, but I'm wondering about these particular methods.
See also: Scala list concatenation, ::: vs ++
The best way to determine the difference between both methods is to look it the source code.
The source of
::
:The source of
+:
:As you can see, for
List
, both methods do one and the same (the compiler will choose List.canBuildFrom for theCanBuildFrom
argument).So, which method to use? Normally one would choose the interface (
+:
) than the implementation (::
) but becauseList
is a general data structure in functional languages it has its own methods which are widely used. Many algorithms are build up the way howList
works. For example you will find a lot of methods which prepend single elements toList
or call the convenienthead
ortail
methods because all these operations areO(1)
. Therefore, if you work locally with aList
(inside of single methods or classes), there is no problem to choose theList
-specific methods. But if you want to communicate between classes, i.e. you want to write some interfaces, you should choose the more generalSeq
interface.