package generic
- Alphabetic
- Public
- All
Type Members
-
trait
IsIterableLike[Repr] extends AnyRef
A trait which can be used to avoid code duplication when defining extension methods that should be applicable both to existing Scala collections (i.e., types extending
Iterable) as well as other (potentially user-defined) types that could be converted to a Scala collection type.A trait which can be used to avoid code duplication when defining extension methods that should be applicable both to existing Scala collections (i.e., types extending
Iterable) as well as other (potentially user-defined) types that could be converted to a Scala collection type. This trait makes it possible to treat Scala collections and types that can be implicitly converted to a collection type uniformly. For example, one can provide extension methods that work both on collection types and onStrings (Strings do not extendIterable, but can be converted toIterable)IsIterableprovides two members:- type member
A, which represents the element type of the targetIterable[A] - value member
conversion, which provides a way to convert between the type we wish to add extension methods to,Repr, andIterable[A].
Usage
One must provide
IsIterableLikeas an implicit parameter type of an implicit conversion. Its usage is shown below. Our objective in the following example is to provide a generic extension methodmapReduceto any type that extends or can be converted toIterable. In our example, this includesString.import scala.collection.Iterable import scala.collection.generic.IsIterableLike class ExtensionMethods[A, Repr](coll: IterableLike[A, Repr]) { def mapReduce[B](mapper: A => B)(reducer: (B, B) => B): B = { val iter = coll.toIterator var res = mapper(iter.next()) while (iter.hasNext) res = reducer(res, mapper(iter.next())) res } } implicit def withExtensions[Repr](coll: Repr)(implicit Iterable: IsIterableLike[Repr]) = new ExtensionMethods(Iterable.conversion(coll)) // See it in action! List(1, 2, 3).mapReduce(_ * 2)(_ + _) // res0: Int = 12 "Yeah, well, you know, that's just, like, your opinion, man.".mapReduce(x => 1)(_ + _) // res1: Int = 59
Here, we begin by creating a class
ExtensionMethodswhich contains ourmapReduceextension method. Note thatExtensionMethodstakes a constructor argumentcollof typeIterableLike[A, Repr], whereArepresents the element type andReprrepresents (typically) the collection type. The implementation ofmapReduceitself is straightforward.The interesting bit is the implicit conversion
withExtensions, which returns an instance ofExtensionMethods. This implicit conversion can only be applied if there is an implicit valueIterableof typeIsIterableLike[Repr]in scope. SinceIsIterableLikeprovides value memberconversion, which gives us a way to convert between whatever type we wish to add an extension method to (in this case,Repr) andIterableLike[A, Repr], we can now convertcollfrom typeReprtoIterableLike[A, Repr]. This allows us to create an instance of theExtensionMethodsclass, which we pass our newIterableLike[A, Repr]to.When the
mapReducemethod is called on some type of which it is not a member, implicit search is triggered. Because implicit conversionwithExtensionsis generic, it will be applied as long as an implicit value of typeIsIterableLike[Repr]can be found. Given thatIsIterableLikecontains implicit members that return values of typeIsIterableLike, this requirement is typically satisfied, and the chain of interactions described in the previous paragraph is set into action. (See theIsIterableLikecompanion object, which contains a precise specification of the available implicits.)Note: Currently, it's not possible to combine the implicit conversion and the class with the extension methods into an implicit class due to limitations of type inference.
Implementing
IsIterableLikefor New TypesOne must simply provide an implicit value of type
IsIterableLikespecific to the new type, or an implicit conversion which returns an instance ofIsIterableLikespecific to the new type.Below is an example of an implementation of the
IsIterableLiketrait where theReprtype isString.implicit val stringRepr: IsIterableLike[String] { type A = Char } = new IsIterableLike[String] { type A = Char val conversion = implicitly[String => IterableOps[Char, Any, String]] }
- type member
Value Members
- object IsIterableLike