Skip to content

Commit

Permalink
add Chain#distinctBy
Browse files Browse the repository at this point in the history
  • Loading branch information
satorg committed Apr 7, 2022
1 parent fca6696 commit 159a315
Show file tree
Hide file tree
Showing 2 changed files with 61 additions and 10 deletions.
65 changes: 55 additions & 10 deletions core/src/main/scala/cats/data/Chain.scala
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,8 @@ import cats.kernel.instances.StaticMethods

import scala.annotation.tailrec
import scala.collection.immutable.SortedMap
import scala.collection.immutable.TreeSet
import scala.collection.immutable.{IndexedSeq => ImIndexedSeq}
import scala.collection.mutable
import scala.collection.mutable.ListBuffer

import Chain.{
Expand Down Expand Up @@ -130,6 +130,13 @@ sealed abstract class Chain[+A] {
*/
final def nonEmpty: Boolean = !isEmpty

// Temporary solution for length comparison in some special cases.
// TODO: Consider implementing more generic `lengthCompare` method.
private def isEmptyOrSingleton: Boolean =
// Chain.Append and Chain.Wrap always contain more than one element.
// TODO: replace with `lengthCompare(1) <= 0`
isEmpty || this.isInstanceOf[Chain.Singleton[_]]

/**
* Concatenates this with `c` in O(1) runtime.
*/
Expand Down Expand Up @@ -690,19 +697,57 @@ sealed abstract class Chain[+A] {

/**
* Remove duplicates. Duplicates are checked using `Order[_]` instance.
*
* Example:
* {{{
* scala> import cats.data.Chain
* scala> val chain = Chain(1, 2, 2, 3)
* scala> chain.distinct
* res0: cats.data.Chain[Int] = Chain(1, 2, 3)
* }}}
*/
def distinct[AA >: A](implicit O: Order[AA]): Chain[AA] = {
implicit val ord: Ordering[AA] = O.toOrdering

var alreadyIn = TreeSet.empty[AA]
if (isEmptyOrSingleton) this
else {
implicit val ord: Ordering[AA] = O.toOrdering

var result = Chain.empty[A]
val seen = mutable.TreeSet.empty[AA]
val it = iterator
while (it.hasNext) {
val next = it.next()
if (seen.add(next))
result = result :+ next
}
result
}
}

foldLeft(Chain.empty[AA]) { (elementsSoFar, b) =>
if (alreadyIn.contains(b)) {
elementsSoFar
} else {
alreadyIn += b
elementsSoFar :+ b
/**
* Remove duplicates by a predicate. Duplicates are checked using `Order[_]` instance.
*
* Example:
* {{{
* scala> import cats.data.Chain
* scala> val chain = Chain(1, 2, 3, 4)
* scala> chain.distinctBy(_ / 2)
* res0: cats.data.Chain[Int] = Chain(1, 2, 4)
* }}}
*/
def distinctBy[B](f: A => B)(implicit O: Order[B]): Chain[A] = {
if (isEmptyOrSingleton) this
else {
implicit val ord: Ordering[B] = O.toOrdering

var result = Chain.empty[A]
val seen = mutable.TreeSet.empty[B]
val it = iterator
while (it.hasNext) {
val next = it.next()
if (seen.add(f(next)))
result = result :+ next
}
result
}
}

Expand Down
6 changes: 6 additions & 0 deletions tests/src/test/scala/cats/tests/ChainSuite.scala
Original file line number Diff line number Diff line change
Expand Up @@ -359,6 +359,12 @@ class ChainSuite extends CatsSuite {
}
}

test("Chain#distinctBy is consistent with List#distinctBy") {
forAll { (a: Chain[Int], f: Int => String) =>
assertEquals(a.distinctBy(f).toList, a.toList.distinctBy(f))
}
}

test("=== is consistent with == (issue #2540)") {
assertEquals(Chain.one(1) |+| Chain.one(2) |+| Chain.one(3), Chain.fromSeq(List(1, 2, 3)))

Expand Down

0 comments on commit 159a315

Please sign in to comment.